signals.py 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213
  1. from django.contrib.auth import get_user_model
  2. from django.db import transaction
  3. from django.db.models.signals import pre_delete
  4. from django.dispatch import Signal, receiver
  5. from misago.categories.models import Category
  6. from misago.categories.signals import delete_category_content, move_category_content
  7. from misago.core.pgutils import chunk_queryset
  8. from misago.core.utils import ANONYMOUS_IP
  9. from misago.users.signals import anonymize_user_content, delete_user_content, username_changed
  10. from .anonymize import ANONYMIZABLE_EVENTS, anonymize_event, anonymize_post_last_likes
  11. from .models import Attachment, Poll, PollVote, Post, PostEdit, PostLike, Thread
  12. delete_post = Signal()
  13. delete_thread = Signal()
  14. merge_post = Signal(providing_args=["other_post"])
  15. merge_thread = Signal(providing_args=["other_thread"])
  16. move_post = Signal()
  17. move_thread = Signal()
  18. @receiver(merge_thread)
  19. def merge_threads(sender, **kwargs):
  20. other_thread = kwargs['other_thread']
  21. other_thread.post_set.update(
  22. category=sender.category,
  23. thread=sender,
  24. )
  25. other_thread.postedit_set.update(
  26. category=sender.category,
  27. thread=sender,
  28. )
  29. other_thread.postlike_set.update(
  30. category=sender.category,
  31. thread=sender,
  32. )
  33. other_thread.subscription_set.exclude(
  34. user__in=sender.subscription_set.values('user'),
  35. ).update(
  36. category=sender.category,
  37. thread=sender,
  38. )
  39. @receiver(merge_post)
  40. def merge_posts(sender, **kwargs):
  41. other_post = kwargs['other_post']
  42. for user in sender.mentions.iterator():
  43. other_post.mentions.add(user)
  44. @receiver(move_thread)
  45. def move_thread_content(sender, **kwargs):
  46. sender.post_set.update(category=sender.category)
  47. sender.postedit_set.update(category=sender.category)
  48. sender.postlike_set.update(category=sender.category)
  49. sender.pollvote_set.update(category=sender.category)
  50. sender.subscription_set.update(category=sender.category)
  51. Poll.objects.filter(thread=sender).update(category=sender.category)
  52. @receiver(delete_category_content)
  53. def delete_category_threads(sender, **kwargs):
  54. sender.subscription_set.all().delete()
  55. sender.pollvote_set.all().delete()
  56. sender.poll_set.all().delete()
  57. sender.postlike_set.all().delete()
  58. sender.thread_set.all().delete()
  59. sender.postedit_set.all().delete()
  60. sender.post_set.all().delete()
  61. @receiver(move_category_content)
  62. def move_category_threads(sender, **kwargs):
  63. new_category = kwargs['new_category']
  64. sender.thread_set.update(category=new_category)
  65. sender.post_set.filter(category=sender).update(category=new_category)
  66. sender.postedit_set.filter(category=sender).update(category=new_category)
  67. sender.postlike_set.filter(category=sender).update(category=new_category)
  68. sender.poll_set.filter(category=sender).update(category=new_category)
  69. sender.pollvote_set.update(category=new_category)
  70. sender.subscription_set.update(category=new_category)
  71. @receiver(delete_user_content)
  72. def delete_user_threads(sender, **kwargs):
  73. recount_categories = set()
  74. recount_threads = set()
  75. for post in sender.liked_post_set.all():
  76. cleaned_likes = list(filter(lambda i: i['id'] != sender.id, post.last_likes))
  77. if cleaned_likes != post.last_likes:
  78. post.last_likes = cleaned_likes
  79. post.save(update_fields=['last_likes'])
  80. for thread in chunk_queryset(sender.thread_set.all()):
  81. recount_categories.add(thread.category_id)
  82. with transaction.atomic():
  83. thread.delete()
  84. for post in chunk_queryset(sender.post_set.all()):
  85. recount_categories.add(post.category_id)
  86. recount_threads.add(post.thread_id)
  87. with transaction.atomic():
  88. post.delete()
  89. if recount_threads:
  90. changed_threads_qs = Thread.objects.filter(id__in=recount_threads)
  91. for thread in chunk_queryset(changed_threads_qs):
  92. thread.synchronize()
  93. thread.save()
  94. if recount_categories:
  95. for category in Category.objects.filter(id__in=recount_categories):
  96. category.synchronize()
  97. category.save()
  98. @receiver(anonymize_user_content)
  99. def anonymize_user(sender, **kwargs):
  100. Post.objects.filter(poster=sender).update(poster_ip=ANONYMOUS_IP)
  101. PostEdit.objects.filter(editor=sender).update(editor_ip=ANONYMOUS_IP)
  102. PostLike.objects.filter(liker=sender).update(liker_ip=ANONYMOUS_IP)
  103. Attachment.objects.filter(uploader=sender).update(uploader_ip=ANONYMOUS_IP)
  104. Poll.objects.filter(poster=sender).update(poster_ip=ANONYMOUS_IP)
  105. PollVote.objects.filter(voter=sender).update(voter_ip=ANONYMOUS_IP)
  106. @receiver(anonymize_user_content)
  107. def anonymize_user_in_events(sender, **kwargs):
  108. queryset = Post.objects.filter(
  109. is_event=True,
  110. event_type__in=ANONYMIZABLE_EVENTS,
  111. event_context__user__id=sender.id,
  112. )
  113. for event in chunk_queryset(queryset):
  114. anonymize_event(sender, event)
  115. @receiver([anonymize_user_content])
  116. def anonymize_user_in_likes(sender, **kwargs):
  117. for post in chunk_queryset(sender.liked_post_set):
  118. anonymize_post_last_likes(sender, post)
  119. @receiver([anonymize_user_content, username_changed])
  120. def update_usernames(sender, **kwargs):
  121. Thread.objects.filter(starter=sender).update(
  122. starter_name=sender.username,
  123. starter_slug=sender.slug,
  124. )
  125. Thread.objects.filter(last_poster=sender).update(
  126. last_poster_name=sender.username,
  127. last_poster_slug=sender.slug,
  128. )
  129. Thread.objects.filter(best_answer_marked_by=sender).update(
  130. best_answer_marked_by_name=sender.username,
  131. best_answer_marked_by_slug=sender.slug,
  132. )
  133. Post.objects.filter(poster=sender).update(
  134. poster_name=sender.username,
  135. )
  136. Post.objects.filter(last_editor=sender).update(
  137. last_editor_name=sender.username,
  138. last_editor_slug=sender.slug,
  139. )
  140. PostEdit.objects.filter(editor=sender).update(
  141. editor_name=sender.username,
  142. editor_slug=sender.slug,
  143. )
  144. PostLike.objects.filter(liker=sender).update(
  145. liker_name=sender.username,
  146. liker_slug=sender.slug,
  147. )
  148. Attachment.objects.filter(uploader=sender).update(
  149. uploader_name=sender.username,
  150. uploader_slug=sender.slug,
  151. )
  152. Poll.objects.filter(poster=sender).update(
  153. poster_name=sender.username,
  154. poster_slug=sender.slug,
  155. )
  156. PollVote.objects.filter(voter=sender).update(
  157. voter_name=sender.username,
  158. voter_slug=sender.slug,
  159. )
  160. @receiver(pre_delete, sender=get_user_model())
  161. def remove_unparticipated_private_threads(sender, **kwargs):
  162. threads_qs = kwargs['instance'].privatethread_set.all()
  163. for thread in chunk_queryset(threads_qs):
  164. if thread.participants.count() == 1:
  165. with transaction.atomic():
  166. thread.delete()