thread.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315
  1. from django.core.exceptions import ObjectDoesNotExist
  2. from django.db import models
  3. from django.db.models import Q
  4. from django.utils import timezone
  5. from django.utils.translation import gettext_lazy as _
  6. from ...conf import settings
  7. from ...core.utils import slugify
  8. class Thread(models.Model):
  9. WEIGHT_DEFAULT = 0
  10. WEIGHT_PINNED = 1
  11. WEIGHT_GLOBAL = 2
  12. WEIGHT_CHOICES = [
  13. (WEIGHT_DEFAULT, _("Don't pin thread")),
  14. (WEIGHT_PINNED, _("Pin thread within category")),
  15. (WEIGHT_GLOBAL, _("Pin thread globally")),
  16. ]
  17. category = models.ForeignKey("misago_categories.Category", on_delete=models.CASCADE)
  18. title = models.CharField(max_length=255)
  19. slug = models.CharField(max_length=255)
  20. replies = models.PositiveIntegerField(default=0, db_index=True)
  21. has_events = models.BooleanField(default=False)
  22. has_poll = models.BooleanField(default=False)
  23. has_reported_posts = models.BooleanField(default=False)
  24. has_open_reports = models.BooleanField(default=False)
  25. has_unapproved_posts = models.BooleanField(default=False)
  26. has_hidden_posts = models.BooleanField(default=False)
  27. started_on = models.DateTimeField(db_index=True)
  28. last_post_on = models.DateTimeField(db_index=True)
  29. first_post = models.ForeignKey(
  30. "misago_threads.Post",
  31. related_name="+",
  32. null=True,
  33. blank=True,
  34. on_delete=models.SET_NULL,
  35. )
  36. starter = models.ForeignKey(
  37. settings.AUTH_USER_MODEL, null=True, blank=True, on_delete=models.SET_NULL
  38. )
  39. starter_name = models.CharField(max_length=255)
  40. starter_slug = models.CharField(max_length=255)
  41. last_post = models.ForeignKey(
  42. "misago_threads.Post",
  43. related_name="+",
  44. null=True,
  45. blank=True,
  46. on_delete=models.SET_NULL,
  47. )
  48. last_post_is_event = models.BooleanField(default=False)
  49. last_poster = models.ForeignKey(
  50. settings.AUTH_USER_MODEL,
  51. related_name="last_poster_set",
  52. null=True,
  53. blank=True,
  54. on_delete=models.SET_NULL,
  55. )
  56. last_poster_name = models.CharField(max_length=255, null=True, blank=True)
  57. last_poster_slug = models.CharField(max_length=255, null=True, blank=True)
  58. weight = models.PositiveIntegerField(default=WEIGHT_DEFAULT)
  59. is_unapproved = models.BooleanField(default=False, db_index=True)
  60. is_hidden = models.BooleanField(default=False)
  61. is_closed = models.BooleanField(default=False)
  62. best_answer = models.ForeignKey(
  63. "misago_threads.Post",
  64. related_name="+",
  65. null=True,
  66. blank=True,
  67. on_delete=models.SET_NULL,
  68. )
  69. best_answer_is_protected = models.BooleanField(default=False)
  70. best_answer_marked_on = models.DateTimeField(null=True, blank=True)
  71. best_answer_marked_by = models.ForeignKey(
  72. settings.AUTH_USER_MODEL,
  73. related_name="marked_best_answer_set",
  74. null=True,
  75. blank=True,
  76. on_delete=models.SET_NULL,
  77. )
  78. best_answer_marked_by_name = models.CharField(max_length=255, null=True, blank=True)
  79. best_answer_marked_by_slug = models.CharField(max_length=255, null=True, blank=True)
  80. participants = models.ManyToManyField(
  81. settings.AUTH_USER_MODEL,
  82. related_name="privatethread_set",
  83. through="ThreadParticipant",
  84. through_fields=("thread", "user"),
  85. )
  86. class Meta:
  87. indexes = [
  88. models.Index(
  89. name="misago_thread_pinned_glob_part",
  90. fields=["weight"],
  91. condition=Q(weight=2),
  92. ),
  93. models.Index(
  94. name="misago_thread_pinned_loca_part",
  95. fields=["weight"],
  96. condition=Q(weight=1),
  97. ),
  98. models.Index(
  99. name="misago_thread_not_pinned_part",
  100. fields=["weight"],
  101. condition=Q(weight=0),
  102. ),
  103. models.Index(
  104. name="misago_thread_not_global_part",
  105. fields=["weight"],
  106. condition=Q(weight__lt=2),
  107. ),
  108. models.Index(
  109. name="misago_thread_has_reporte_part",
  110. fields=["has_reported_posts"],
  111. condition=Q(has_reported_posts=True),
  112. ),
  113. models.Index(
  114. name="misago_thread_has_unappro_part",
  115. fields=["has_unapproved_posts"],
  116. condition=Q(has_unapproved_posts=True),
  117. ),
  118. models.Index(
  119. name="misago_thread_is_visible_part",
  120. fields=["is_hidden"],
  121. condition=Q(is_hidden=False),
  122. ),
  123. ]
  124. index_together = [
  125. ["category", "id"],
  126. ["category", "last_post_on"],
  127. ["category", "replies"],
  128. ]
  129. def __str__(self):
  130. return self.title
  131. def delete(self, *args, **kwargs):
  132. from ..signals import delete_thread
  133. delete_thread.send(sender=self)
  134. super().delete(*args, **kwargs)
  135. def merge(self, other_thread):
  136. if self.pk == other_thread.pk:
  137. raise ValueError("thread can't be merged with itself")
  138. from ..signals import merge_thread
  139. merge_thread.send(sender=self, other_thread=other_thread)
  140. def move(self, new_category):
  141. from ..signals import move_thread
  142. self.category = new_category
  143. move_thread.send(sender=self)
  144. def synchronize(self):
  145. try:
  146. self.has_poll = bool(self.poll)
  147. except ObjectDoesNotExist:
  148. self.has_poll = False
  149. self.replies = self.post_set.filter(is_event=False, is_unapproved=False).count()
  150. if self.replies > 0:
  151. self.replies -= 1
  152. reported_post_qs = self.post_set.filter(has_reports=True)
  153. self.has_reported_posts = reported_post_qs.exists()
  154. if self.has_reported_posts:
  155. open_reports_qs = self.post_set.filter(has_open_reports=True)
  156. self.has_open_reports = open_reports_qs.exists()
  157. else:
  158. self.has_open_reports = False
  159. unapproved_post_qs = self.post_set.filter(is_unapproved=True)
  160. self.has_unapproved_posts = unapproved_post_qs.exists()
  161. hidden_post_qs = self.post_set.filter(is_hidden=True)[:1]
  162. self.has_hidden_posts = hidden_post_qs.exists()
  163. posts = self.post_set.order_by("id")
  164. first_post = posts.first()
  165. self.set_first_post(first_post)
  166. last_post = posts.filter(is_unapproved=False).last()
  167. if last_post:
  168. self.set_last_post(last_post)
  169. else:
  170. self.set_last_post(first_post)
  171. self.has_events = False
  172. if last_post:
  173. if last_post.is_event:
  174. self.has_events = True
  175. else:
  176. self.has_events = self.post_set.filter(is_event=True).exists()
  177. @property
  178. def has_best_answer(self):
  179. return bool(self.best_answer_id)
  180. @property
  181. def replies_in_ks(self):
  182. return "%sK" % round(self.replies / 1000, 0)
  183. @property
  184. def thread_type(self):
  185. return self.category.thread_type
  186. def get_api_url(self):
  187. return self.thread_type.get_thread_api_url(self)
  188. def get_editor_api_url(self):
  189. return self.thread_type.get_thread_editor_api_url(self)
  190. def get_merge_api_url(self):
  191. return self.thread_type.get_thread_merge_api_url(self)
  192. def get_posts_api_url(self):
  193. return self.thread_type.get_thread_posts_api_url(self)
  194. def get_post_merge_api_url(self):
  195. return self.thread_type.get_post_merge_api_url(self)
  196. def get_post_move_api_url(self):
  197. return self.thread_type.get_post_move_api_url(self)
  198. def get_post_split_api_url(self):
  199. return self.thread_type.get_post_split_api_url(self)
  200. def get_poll_api_url(self):
  201. return self.thread_type.get_thread_poll_api_url(self)
  202. def get_absolute_url(self, page=1):
  203. return self.thread_type.get_thread_absolute_url(self, page)
  204. def get_new_post_url(self):
  205. return self.thread_type.get_thread_new_post_url(self)
  206. def get_last_post_url(self):
  207. return self.thread_type.get_thread_last_post_url(self)
  208. def get_best_answer_url(self):
  209. return self.thread_type.get_thread_best_answer_url(self)
  210. def get_unapproved_post_url(self):
  211. return self.thread_type.get_thread_unapproved_post_url(self)
  212. def set_title(self, title):
  213. self.title = title
  214. self.slug = slugify(title)
  215. def set_first_post(self, post):
  216. self.started_on = post.posted_on
  217. self.first_post = post
  218. self.starter = post.poster
  219. self.starter_name = post.poster_name
  220. if post.poster:
  221. self.starter_slug = post.poster.slug
  222. else:
  223. self.starter_slug = slugify(post.poster_name)
  224. self.is_unapproved = post.is_unapproved
  225. self.is_hidden = post.is_hidden
  226. def set_last_post(self, post):
  227. self.last_post_on = post.posted_on
  228. self.last_post_is_event = post.is_event
  229. self.last_post = post
  230. self.last_poster = post.poster
  231. self.last_poster_name = post.poster_name
  232. if post.poster:
  233. self.last_poster_slug = post.poster.slug
  234. else:
  235. self.last_poster_slug = slugify(post.poster_name)
  236. def set_best_answer(self, user, post):
  237. if post.thread_id != self.id:
  238. raise ValueError("post to set as best answer must be in same thread")
  239. if post.is_first_post:
  240. raise ValueError("post to set as best answer can't be first post")
  241. if post.is_hidden:
  242. raise ValueError("post to set as best answer can't be hidden")
  243. if post.is_unapproved:
  244. raise ValueError("post to set as best answer can't be unapproved")
  245. self.best_answer = post
  246. self.best_answer_is_protected = post.is_protected
  247. self.best_answer_marked_on = timezone.now()
  248. self.best_answer_marked_by = user
  249. self.best_answer_marked_by_name = user.username
  250. self.best_answer_marked_by_slug = user.slug
  251. def clear_best_answer(self):
  252. self.best_answer = None
  253. self.best_answer_is_protected = False
  254. self.best_answer_marked_on = None
  255. self.best_answer_marked_by = None
  256. self.best_answer_marked_by_name = None
  257. self.best_answer_marked_by_slug = None