thread.py 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  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 thread_type(self):
  182. return self.category.thread_type
  183. def get_api_url(self):
  184. return self.thread_type.get_thread_api_url(self)
  185. def get_editor_api_url(self):
  186. return self.thread_type.get_thread_editor_api_url(self)
  187. def get_merge_api_url(self):
  188. return self.thread_type.get_thread_merge_api_url(self)
  189. def get_posts_api_url(self):
  190. return self.thread_type.get_thread_posts_api_url(self)
  191. def get_post_merge_api_url(self):
  192. return self.thread_type.get_post_merge_api_url(self)
  193. def get_post_move_api_url(self):
  194. return self.thread_type.get_post_move_api_url(self)
  195. def get_post_split_api_url(self):
  196. return self.thread_type.get_post_split_api_url(self)
  197. def get_poll_api_url(self):
  198. return self.thread_type.get_thread_poll_api_url(self)
  199. def get_absolute_url(self, page=1):
  200. return self.thread_type.get_thread_absolute_url(self, page)
  201. def get_new_post_url(self):
  202. return self.thread_type.get_thread_new_post_url(self)
  203. def get_last_post_url(self):
  204. return self.thread_type.get_thread_last_post_url(self)
  205. def get_best_answer_url(self):
  206. return self.thread_type.get_thread_best_answer_url(self)
  207. def get_unapproved_post_url(self):
  208. return self.thread_type.get_thread_unapproved_post_url(self)
  209. def set_title(self, title):
  210. self.title = title
  211. self.slug = slugify(title)
  212. def set_first_post(self, post):
  213. self.started_on = post.posted_on
  214. self.first_post = post
  215. self.starter = post.poster
  216. self.starter_name = post.poster_name
  217. if post.poster:
  218. self.starter_slug = post.poster.slug
  219. else:
  220. self.starter_slug = slugify(post.poster_name)
  221. self.is_unapproved = post.is_unapproved
  222. self.is_hidden = post.is_hidden
  223. def set_last_post(self, post):
  224. self.last_post_on = post.posted_on
  225. self.last_post_is_event = post.is_event
  226. self.last_post = post
  227. self.last_poster = post.poster
  228. self.last_poster_name = post.poster_name
  229. if post.poster:
  230. self.last_poster_slug = post.poster.slug
  231. else:
  232. self.last_poster_slug = slugify(post.poster_name)
  233. def set_best_answer(self, user, post):
  234. if post.thread_id != self.id:
  235. raise ValueError("post to set as best answer must be in same thread")
  236. if post.is_first_post:
  237. raise ValueError("post to set as best answer can't be first post")
  238. if post.is_hidden:
  239. raise ValueError("post to set as best answer can't be hidden")
  240. if post.is_unapproved:
  241. raise ValueError("post to set as best answer can't be unapproved")
  242. self.best_answer = post
  243. self.best_answer_is_protected = post.is_protected
  244. self.best_answer_marked_on = timezone.now()
  245. self.best_answer_marked_by = user
  246. self.best_answer_marked_by_name = user.username
  247. self.best_answer_marked_by_slug = user.slug
  248. def clear_best_answer(self):
  249. self.best_answer = None
  250. self.best_answer_is_protected = False
  251. self.best_answer_marked_on = None
  252. self.best_answer_marked_by = None
  253. self.best_answer_marked_by_name = None
  254. self.best_answer_marked_by_slug = None