list.py 3.0 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. from datetime import timedelta
  2. from django.contrib.auth import get_user_model
  3. from django.db.models import Count
  4. from django.utils import timezone
  5. from rest_framework.pagination import PageNumberPagination
  6. from misago.conf import settings
  7. from misago.core.cache import cache
  8. from misago.core.shortcuts import get_object_or_404
  9. from misago.forums.models import Forum
  10. from misago.users.online.utils import get_online_queryset
  11. from misago.users.permissions.profiles import allow_see_users_online_list
  12. from misago.users.models import Rank
  13. from misago.users.serializers import OnlineUserSerializer, ScoredUserSerializer
  14. def active(request, queryset):
  15. cache_key = 'misago_active_posters_ranking'
  16. ranking = cache.get(cache_key, False)
  17. if ranking is False:
  18. ranking = real_active()
  19. cache.set(cache_key, ranking, 18*3600)
  20. return ranking
  21. def real_active():
  22. tracked_period = settings.MISAGO_RANKING_LENGTH
  23. tracked_since = timezone.now() - timedelta(days=tracked_period)
  24. ranked_forums = [forum.pk for forum in Forum.objects.all_forums()]
  25. User = get_user_model()
  26. queryset = User.objects.filter(posts__gt=0)
  27. queryset = queryset.filter(post__posted_on__gte=tracked_since,
  28. post__forum__in=ranked_forums)
  29. queryset = queryset.annotate(num_posts=Count('post'))
  30. queryset = queryset.select_related('user__rank')
  31. queryset = queryset.order_by('-num_posts', 'slug')
  32. users_ranking = []
  33. for result in queryset[:settings.MISAGO_RANKING_SIZE]:
  34. result.score = result.num_posts
  35. users_ranking.append(result)
  36. return {'data': ScoredUserSerializer(users_ranking, many=True).data}
  37. def online(request, queryset):
  38. allow_see_users_online_list(request.user)
  39. cache_key = 'users_online_cache_%s' % request.user.acl_key
  40. online_list = cache.get(cache_key, False)
  41. if online_list is False:
  42. online_list = real_online(request)
  43. cache.set(cache_key, online_list, settings.MISAGO_ONLINE_LIST_CACHE)
  44. return online_list
  45. def real_online(request):
  46. queryset = get_online_queryset(request.user).order_by('last_click')
  47. queryset = queryset[:settings.MISAGO_ONLINE_LIST_SIZE]
  48. users_online = []
  49. for result in queryset:
  50. result.user.last_click = result.last_click
  51. users_online.append(result.user)
  52. return {'data': OnlineUserSerializer(users_online, many=True).data}
  53. def rank(request, queryset):
  54. rank_slug = request.query_params.get('rank')
  55. if not rank_slug:
  56. return
  57. rank = get_object_or_404(Rank.objects.filter(is_tab=True), slug=rank_slug)
  58. queryset = queryset.filter(rank=rank).order_by('slug')
  59. return {'queryset': queryset, 'paginate': True}
  60. LISTS = {
  61. 'active': active,
  62. 'online': online,
  63. 'rank': rank,
  64. }
  65. def list_endpoint(request, queryset):
  66. list_type = request.query_params.get('list')
  67. list_handler = LISTS.get(list_type)
  68. if list_handler:
  69. return list_handler(request, queryset)
  70. else:
  71. return