lists.py 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. from datetime import timedelta
  2. from django.conf import settings
  3. from django.contrib.auth import get_user_model
  4. from django.core.urlresolvers import reverse
  5. from django.db.models import Count
  6. from django.http import Http404
  7. from django.shortcuts import redirect, render as django_render
  8. from django.utils import timezone
  9. from misago.forums.models import Forum
  10. from misago.core.cache import cache
  11. from misago.core.shortcuts import get_object_or_404, paginate
  12. from misago.core.utils import format_plaintext_for_html
  13. from misago.users.models import Rank
  14. from misago.users.pages import users_list
  15. from misago.users.permissions.profiles import allow_browse_users_list
  16. from misago.users.serializers import ScoredUserSerializer
  17. def render(request, template, context):
  18. request.frontend_context['USERS_LISTS'] = []
  19. context['pages'] = users_list.get_sections(request)
  20. for page in context['pages']:
  21. page['reversed_link'] = reverse(page['link'])
  22. request.frontend_context['USERS_LISTS'].append({
  23. 'name': unicode(page['name']),
  24. 'component': page['component'],
  25. })
  26. active_rank = context.get('rank')
  27. for rank in Rank.objects.filter(is_tab=True).order_by('order'):
  28. context['pages'].append({
  29. 'name': rank.name,
  30. 'reversed_link': reverse('misago:users_rank',
  31. kwargs={'rank_slug': rank.slug}),
  32. 'is_active': active_rank.pk == rank.pk if active_rank else None
  33. })
  34. if rank.description:
  35. description = {
  36. 'plain': rank.description,
  37. 'html': format_plaintext_for_html(rank.description)
  38. }
  39. else:
  40. description = None
  41. request.frontend_context['USERS_LISTS'].append({
  42. 'name': rank.name,
  43. 'slug': rank.slug,
  44. 'css_class': rank.css_class,
  45. 'description': description,
  46. 'component': 'rank',
  47. })
  48. for page in context['pages']:
  49. if page['is_active']:
  50. context['active_page'] = page
  51. break
  52. return django_render(request, template, context)
  53. def allow_see_list(f):
  54. def decorator(request, *args, **kwargs):
  55. allow_browse_users_list(request.user)
  56. return f(request, *args, **kwargs)
  57. return decorator
  58. @allow_see_list
  59. def lander(request):
  60. default = users_list.get_default_link()
  61. return redirect(default)
  62. def list_view(request, template, queryset, page, context=None):
  63. context = context or {}
  64. context['users'] = paginate(queryset, page, 6 * 3, 5)
  65. return render(request, template, context)
  66. @allow_see_list
  67. def active_posters(request):
  68. ranking = get_active_posters_rankig()
  69. request.frontend_context['USERS'] = {
  70. 'tracked_period': settings.MISAGO_RANKING_LENGTH,
  71. 'results': ScoredUserSerializer(ranking['users'], many=True).data,
  72. 'count': ranking['users_count']
  73. }
  74. template = "misago/userslists/active_posters.html"
  75. return render(request, template, {
  76. 'tracked_period': settings.MISAGO_RANKING_LENGTH,
  77. 'users': ranking['users'],
  78. 'users_count': ranking['users_count']
  79. })
  80. def get_active_posters_rankig():
  81. cache_key = 'misago_active_posters_ranking'
  82. ranking = cache.get(cache_key, 'nada')
  83. if ranking == 'nada':
  84. ranking = get_real_active_posts_ranking()
  85. cache.set(cache_key, ranking, 18*3600)
  86. return ranking
  87. def get_real_active_posts_ranking():
  88. tracked_period = settings.MISAGO_RANKING_LENGTH
  89. tracked_since = timezone.now() - timedelta(days=tracked_period)
  90. ranked_forums = [forum.pk for forum in Forum.objects.all_forums()]
  91. User = get_user_model()
  92. queryset = User.objects.filter(posts__gt=0)
  93. queryset = queryset.filter(post__posted_on__gte=tracked_since,
  94. post__forum__in=ranked_forums)
  95. queryset = queryset.annotate(score=Count('post'))
  96. queryset = queryset.select_related('user__rank')
  97. queryset = queryset.order_by('-score')
  98. queryset = queryset[:settings.MISAGO_RANKING_SIZE]
  99. return {
  100. 'users': [user for user in queryset],
  101. 'users_count': queryset.count()
  102. }
  103. @allow_see_list
  104. def rank(request, rank_slug, page=0):
  105. rank = get_object_or_404(Rank.objects.filter(is_tab=True), slug=rank_slug)
  106. queryset = rank.user_set.order_by('slug')
  107. template = "misago/userslists/rank.html"
  108. return list_view(request, template, queryset, page, {'rank': rank})