thread.py 9.8 KB

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