jumps.py 16 KB

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