jumps.py 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. from django.core.urlresolvers import reverse
  2. from django.db import transaction
  3. from django.shortcuts import redirect
  4. from django.utils import timezone
  5. from django.utils.translation import ugettext as _
  6. from misago import messages
  7. from misago.acl.exceptions import ACLError403, ACLError404
  8. from misago.apps.errors import error403, error404
  9. from misago.conf import settings
  10. from misago.decorators import block_guest, check_csrf
  11. from misago.markdown import post_markdown
  12. from misago.models import Forum, Checkpoint, Thread, Post, Karma, User, WatchedThread
  13. from misago.monitor import monitor, UpdatingMonitor
  14. from misago.readstrackers import ThreadsTracker
  15. from misago.utils.strings import short_string, slugify
  16. from misago.utils.views import json_response
  17. from misago.apps.threadtype.base import ViewBase
  18. class JumpView(ViewBase):
  19. def fetch_thread(self, thread):
  20. self.thread = Thread.objects.get(pk=thread)
  21. self.forum = self.thread.forum
  22. self.request.acl.forums.allow_forum_view(self.forum)
  23. self.request.acl.threads.allow_thread_view(self.request.user, self.thread)
  24. def fetch_post(self, post):
  25. self.post = self.thread.post_set.get(pk=post)
  26. self.request.acl.threads.allow_post_jump(self.request.user, self.thread, self.post)
  27. def make_jump(self):
  28. raise NotImplementedError('JumpView cannot be called directly.')
  29. def __call__(self, request, slug=None, thread=None, post=None):
  30. self.request = request
  31. self.parents = []
  32. try:
  33. self._type_available()
  34. self.fetch_thread(thread)
  35. if self.forum.level:
  36. self.parents = Forum.objects.forum_parents(self.forum.pk, True)
  37. self.check_forum_type()
  38. self._check_permissions()
  39. if post:
  40. self.fetch_post(post)
  41. return self.make_jump()
  42. except (Thread.DoesNotExist, Post.DoesNotExist):
  43. return error404(self.request)
  44. except ACLError403 as e:
  45. return error403(request, e)
  46. except ACLError404 as e:
  47. return error404(request, e)
  48. class LastReplyBaseView(JumpView):
  49. def make_jump(self):
  50. return self.redirect_to_post(self.thread.post_set.order_by('-id')[:1][0])
  51. class FindReplyBaseView(JumpView):
  52. def make_jump(self):
  53. return self.redirect_to_post(self.post)
  54. class NewReplyBaseView(JumpView):
  55. def make_jump(self):
  56. if not self.request.user.is_authenticated():
  57. return self.redirect_to_post(self.thread.post_set.order_by('-id')[:1][0])
  58. tracker = ThreadsTracker(self.request, self.forum)
  59. read_date = tracker.read_date(self.thread)
  60. post = self.thread.post_set.filter(date__gt=read_date).order_by('id')[:1]
  61. if not post:
  62. return self.redirect_to_post(self.thread.post_set.order_by('-id')[:1][0])
  63. return self.redirect_to_post(post[0])
  64. class FirstModeratedBaseView(JumpView):
  65. def make_jump(self):
  66. if not self.request.acl.threads.can_approve(self.forum):
  67. raise ACLError404()
  68. try:
  69. return self.redirect_to_post(
  70. self.thread.post_set.filter(moderated=True)[:1][0])
  71. except IndexError:
  72. return error404(self.request)
  73. class FirstReportedBaseView(JumpView):
  74. def make_jump(self):
  75. if not self.request.acl.threads.can_mod_posts(self.forum):
  76. raise ACLError404()
  77. try:
  78. return self.redirect_to_post(
  79. self.thread.post_set.filter(reported=True)[:1][0])
  80. except Post.DoesNotExist:
  81. return error404(self.request)
  82. class ShowHiddenRepliesBaseView(JumpView):
  83. def make_jump(self):
  84. @block_guest
  85. @check_csrf
  86. def view(request):
  87. ignored_exclusions = request.session.get('unignore_threads', [])
  88. ignored_exclusions.append(self.thread.pk)
  89. request.session['unignore_threads'] = ignored_exclusions
  90. messages.success(request, _('Replies made to this thread by members on your ignore list have been revealed.'), 'threads')
  91. return redirect(reverse(self.type_prefix, kwargs={'thread': self.thread.pk, 'slug': self.thread.slug}))
  92. return view(self.request)
  93. class WatchThreadBaseView(JumpView):
  94. def get_retreat(self):
  95. return redirect(self.request.POST.get('retreat', reverse('thread', kwargs={'thread': self.thread.pk, 'slug': self.thread.slug})))
  96. def update_watcher(self, request, watcher):
  97. messages.success(request, _('This thread has been added to your watched threads list.'), 'threads')
  98. def make_jump(self):
  99. @block_guest
  100. @check_csrf
  101. def view(request):
  102. try:
  103. watcher = WatchedThread.objects.get(user=request.user, thread=self.thread)
  104. except WatchedThread.DoesNotExist:
  105. watcher = WatchedThread()
  106. watcher.user = request.user
  107. watcher.forum = self.forum
  108. watcher.thread = self.thread
  109. watcher.starter_id = self.thread.start_poster_id
  110. watcher.last_read = timezone.now()
  111. watcher.deleted = False
  112. self.update_watcher(request, watcher)
  113. if not watcher.deleted:
  114. if watcher.pk:
  115. watcher.save(force_update=True)
  116. else:
  117. watcher.save(force_insert=True)
  118. return self.get_retreat()
  119. return view(self.request)
  120. class WatchEmailThreadBaseView(WatchThreadBaseView):
  121. def update_watcher(self, request, watcher):
  122. watcher.email = True
  123. if watcher.pk:
  124. messages.success(request, _('You will now receive e-mail with notification when somebody replies to this thread.'), 'threads')
  125. else:
  126. messages.success(request, _('This thread has been added to your watched threads list. You will also receive e-mail with notification when somebody replies to it.'), 'threads')
  127. class UnwatchThreadBaseView(WatchThreadBaseView):
  128. def update_watcher(self, request, watcher):
  129. watcher.deleted = True
  130. if watcher.pk:
  131. watcher.delete()
  132. if watcher.email:
  133. messages.success(request, _('This thread has been removed from your watched threads list. You will no longer receive e-mails with notifications when somebody replies to it.'), 'threads')
  134. else:
  135. messages.success(request, _('This thread has been removed from your watched threads list.'), 'threads')
  136. class UnwatchEmailThreadBaseView(WatchThreadBaseView):
  137. def update_watcher(self, request, watcher):
  138. watcher.email = False
  139. messages.success(request, _('You will no longer receive e-mails with notifications when somebody replies to this thread.'), 'threads')
  140. class UpvotePostBaseView(JumpView):
  141. def make_jump(self):
  142. @block_guest
  143. @check_csrf
  144. @transaction.commit_on_success
  145. def view(request):
  146. if self.post.user_id == request.user.id:
  147. return error404(request)
  148. self.check_acl(request)
  149. user = User.objects.block_user(request.user)
  150. try:
  151. vote = Karma.objects.get(user=request.user, post=self.post)
  152. if self.thread.start_post_id == self.post.pk:
  153. if vote.score > 0:
  154. self.thread.upvotes -= 1
  155. else:
  156. self.thread.downvotes -= 1
  157. if vote.score > 0:
  158. self.post.upvotes -= 1
  159. user.karma_given_p -= 1
  160. if self.post.user_id:
  161. self.post.user.karma_p -= 1
  162. else:
  163. self.post.downvotes -= 1
  164. user.karma_given_n -= 1
  165. if self.post.user_id:
  166. self.post.user.karma_n -= 1
  167. except Karma.DoesNotExist:
  168. vote = Karma()
  169. vote.forum = self.forum
  170. vote.thread = self.thread
  171. vote.post = self.post
  172. vote.user = request.user
  173. vote.user_name = request.user.username
  174. vote.user_slug = request.user.username_slug
  175. vote.date = timezone.now()
  176. vote.ip = request.session.get_ip(request)
  177. vote.agent = request.META.get('HTTP_USER_AGENT')
  178. self.make_vote(request, vote)
  179. vote.save()
  180. if self.thread.start_post_id == self.post.pk:
  181. if vote.score > 0:
  182. self.thread.upvotes += 1
  183. else:
  184. self.thread.downvotes += 1
  185. self.thread.save(force_update=True)
  186. if vote.score > 0:
  187. self.post.upvotes += 1
  188. user.karma_given_p += 1
  189. if self.post.user_id:
  190. self.post.user.karma_p += 1
  191. self.post.user.score += settings.score_reward_karma_positive
  192. else:
  193. self.post.downvotes += 1
  194. user.karma_given_n += 1
  195. if self.post.user_id:
  196. self.post.user.karma_n += 1
  197. self.post.user.score -= settings.score_reward_karma_negative
  198. self.post.save(force_update=True)
  199. user.last_vote = timezone.now()
  200. if self.post.user_id:
  201. self.post.user.save(force_update=True)
  202. user.save(force_update=True)
  203. if request.is_ajax():
  204. return json_response(request, {
  205. 'score_total': self.post.upvotes - self.post.downvotes,
  206. 'score_upvotes': self.post.upvotes,
  207. 'score_downvotes': self.post.downvotes,
  208. 'user_vote': vote.score,
  209. })
  210. messages.success(request, _('Your vote has been saved.'), 'threads_%s' % self.post.pk)
  211. return self.redirect_to_post(self.post)
  212. return view(self.request)
  213. def check_acl(self, request):
  214. request.acl.threads.allow_post_upvote(self.forum)
  215. def make_vote(self, request, vote):
  216. vote.score = 1
  217. class DownvotePostBaseView(UpvotePostBaseView):
  218. def check_acl(self, request):
  219. request.acl.threads.allow_post_downvote(self.forum)
  220. def make_vote(self, request, vote):
  221. vote.score = -1
  222. class ReportPostBaseView(JumpView):
  223. def make_jump(self):
  224. self.request.acl.reports.allow_report()
  225. @block_guest
  226. @check_csrf
  227. def view(request):
  228. report = None
  229. made_report = False
  230. if self.post.reported:
  231. report = self.post.live_report()
  232. if report and report.start_poster_id != request.user.pk:
  233. # Append our q.q to existing report?
  234. try:
  235. report.checkpoint_set.get(user=request.user, action="reported")
  236. except Checkpoint.DoesNotExist:
  237. report.set_checkpoint(request, 'reported', request.useruser)
  238. self.post.add_reporter(request.user)
  239. self.post.save(force_update=True)
  240. made_report = True
  241. if not report:
  242. # File up new report
  243. now = timezone.now()
  244. reason_post = _('''
  245. Member @%(reporter)s has reported following post by @%(reported)s:
  246. %(quote)s
  247. **Post link:** <%(post)s>
  248. ''')
  249. reason_post = reason_post.strip() % {
  250. 'reporter': request.user.username,
  251. 'reported': self.post.user_name,
  252. 'post': settings.BOARD_ADDRESS + self.redirect_to_post(self.post)['Location'],
  253. 'quote': self.post.quote(),
  254. }
  255. md, reason_post_preparsed = post_markdown(reason_post)
  256. reports = Forum.objects.special_model('reports')
  257. report = Thread.objects.create(
  258. forum=reports,
  259. weight=2,
  260. name=self.thread.name,
  261. slug=slugify(self.thread.slug),
  262. start=now,
  263. start_poster=request.user,
  264. start_poster_name=request.user.username,
  265. start_poster_slug=request.user.username_slug,
  266. start_poster_style=request.user.rank.style,
  267. last=now,
  268. last_poster=request.user,
  269. last_poster_name=request.user.username,
  270. last_poster_slug=request.user.username_slug,
  271. last_poster_style=request.user.rank.style,
  272. report_for=self.post,
  273. )
  274. reason = Post.objects.create(
  275. forum=reports,
  276. thread=report,
  277. user=request.user,
  278. user_name=request.user.username,
  279. ip=request.session.get_ip(self.request),
  280. agent=request.META.get('HTTP_USER_AGENT'),
  281. post=reason_post,
  282. post_preparsed=reason_post_preparsed,
  283. date=now,
  284. )
  285. report.start_post = reason
  286. report.last_post = reason
  287. report.save(force_update=True)
  288. for m in self.post.mentions.all():
  289. reason.mentions.add(m)
  290. self.post.reported = True
  291. self.post.add_reporter(self.request.user)
  292. self.post.save(force_update=True)
  293. self.thread.replies_reported += 1
  294. self.thread.save(force_update=True)
  295. with UpdatingMonitor() as cm:
  296. monitor.increase('reported_posts')
  297. made_report = True
  298. if made_report:
  299. if request.is_ajax():
  300. return json_response(request, message=_("Selected post has been reported and will receive moderator attention. Thank you."))
  301. messages.info(request, _("Selected post has been reported and will receive moderator attention. Thank you."), 'threads_%s' % self.post.pk)
  302. else:
  303. if request.is_ajax():
  304. return json_response(request, message=_("You have already reported this post. One of moderators will handle it as soon as it is possible. Thank you for your patience."))
  305. messages.info(request, _("You have already reported this post. One of moderators will handle it as soon as it is possible. Thank you for your patience."), 'threads_%s' % self.post.pk)
  306. return self.redirect_to_post(self.post)
  307. return view(self.request)
  308. class ShowPostReportBaseView(JumpView):
  309. def make_jump(self):
  310. self.request.acl.reports.allow_report()
  311. @block_guest
  312. def view(request):
  313. if not self.post.reported:
  314. return error404(request)
  315. reports = Forum.objects.special_model('reports')
  316. self.request.acl.forums.allow_forum_view(reports)
  317. report = self.post.live_report()
  318. if not report:
  319. return error404(request)
  320. return redirect(reverse('report', kwargs={'thread': report.pk, 'slug': report.slug}))
  321. return view(self.request)