thread.py 9.7 KB

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