views.py 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. import json
  2. import logging
  3. from datetime import datetime
  4. from celery import states
  5. from django.conf import settings
  6. from django.contrib.auth.decorators import login_required
  7. from django.contrib.auth.mixins import LoginRequiredMixin
  8. from django.core.cache import cache
  9. from django.db.models import Avg, Count, F
  10. from django.http import HttpResponse
  11. from django.views.generic import DetailView, ListView
  12. from django.views.generic.list import MultipleObjectMixin
  13. from django_celery_results.models import TaskResult
  14. from games.tasks import update_roms
  15. from .models import Developer, Game, GameCollection, GameSystem, Genre, Publisher
  16. logger = logging.Logger(__name__)
  17. IN_PROGRESS_STATES = [states.PENDING, states.STARTED, states.RETRY]
  18. class RecentGameList(ListView):
  19. model = Game
  20. paginate_by = 10
  21. queryset = Game.objects.order_by("-created")[:70]
  22. def get_context_data(self, **kwargs):
  23. cached_game_id = cache.get("todays_game_id", None)
  24. if not cached_game_id:
  25. todays_game = (
  26. Game.objects.filter(rating__gte=settings.FEATURED_THRESHOLD, featured_on__isnull=True)
  27. .order_by("?")
  28. .first()
  29. )
  30. featured_collection = GameCollection.objects.filter(
  31. slug="emus-featured"
  32. ).first()
  33. if featured_collection:
  34. featured_collection.games.add(todays_game)
  35. featured_collection.save()
  36. cache.set("todays_game_id", todays_game.id, settings.FEATURED_GAME_DURATION)
  37. todays_game.featured_on = datetime.now()
  38. todays_game.save(update_fields=["featured_on"])
  39. else:
  40. todays_game = Game.objects.get(id=cached_game_id)
  41. return super(RecentGameList, self).get_context_data(
  42. todays_game=todays_game,
  43. **kwargs,
  44. )
  45. class LibraryGameList(ListView):
  46. template_name = "games/game_library_list.html"
  47. model = Game
  48. paginate_by = 400
  49. default_sort_key = "-created"
  50. def get_context_data(self, **kwargs):
  51. game_system_slug = self.request.GET.get("game_system")
  52. order_by = self.request.GET.get("order_by", self.default_sort_key)
  53. if order_by[0] == "-":
  54. order_by = order_by[1:]
  55. object_list = Game.objects.order_by(F(order_by).desc(nulls_last=True))
  56. else:
  57. object_list = Game.objects.order_by(F(order_by).asc(nulls_last=True))
  58. if game_system_slug:
  59. object_list = object_list.filter(
  60. game_system__retropie_slug=game_system_slug
  61. )
  62. context = super(LibraryGameList, self).get_context_data(
  63. object_list=object_list, **kwargs
  64. )
  65. return context
  66. class FilterableBaseListView(ListView):
  67. def get_queryset(self, **kwargs):
  68. order_by = self.request.GET.get("order_by", "name")
  69. queryset = super().get_queryset(**kwargs)
  70. queryset = queryset.annotate(num_games=Count("game")).annotate(
  71. rating=Avg("game__rating")
  72. )
  73. if order_by[0] == "-":
  74. order_by = order_by[1:]
  75. queryset = queryset.order_by(F(order_by).desc(nulls_last=True))
  76. else:
  77. queryset = queryset.order_by(F(order_by).asc(nulls_last=True))
  78. return queryset
  79. class PublisherList(FilterableBaseListView):
  80. model = Publisher
  81. class DeveloperList(FilterableBaseListView):
  82. model = Developer
  83. class GenreList(FilterableBaseListView):
  84. model = Genre
  85. class GameDetail(DetailView):
  86. model = Game
  87. class GamePlayDetail(DetailView, LoginRequiredMixin):
  88. template_name = "games/game_play_detail.html"
  89. model = Game
  90. class GameSystemList(ListView):
  91. model = GameSystem
  92. class GameSystemDetail(DetailView, MultipleObjectMixin):
  93. model = GameSystem
  94. paginate_by = 20
  95. def get_context_data(self, **kwargs):
  96. object_list = Game.objects.filter(game_system=self.get_object())
  97. context = super(GameSystemDetail, self).get_context_data(
  98. object_list=object_list, **kwargs
  99. )
  100. return context
  101. class GenreDetail(DetailView, MultipleObjectMixin):
  102. model = Genre
  103. paginate_by = 20
  104. def get_context_data(self, **kwargs):
  105. object_list = Game.objects.filter(genre=self.get_object())
  106. context = super(GenreDetail, self).get_context_data(
  107. object_list=object_list, **kwargs
  108. )
  109. return context
  110. class PublisherDetail(DetailView, MultipleObjectMixin):
  111. model = Publisher
  112. paginate_by = 20
  113. def get_context_data(self, **kwargs):
  114. object_list = Game.objects.filter(publisher=self.get_object())
  115. context = super(PublisherDetail, self).get_context_data(
  116. object_list=object_list, **kwargs
  117. )
  118. return context
  119. class DeveloperDetail(DetailView, MultipleObjectMixin):
  120. model = Developer
  121. paginate_by = 20
  122. def get_context_data(self, **kwargs):
  123. object_list = Game.objects.filter(developer=self.get_object())
  124. context = super(DeveloperDetail, self).get_context_data(
  125. object_list=object_list, **kwargs
  126. )
  127. return context
  128. class GameCollectionList(ListView):
  129. model = GameCollection
  130. class GameCollectionDetail(DetailView, LoginRequiredMixin):
  131. model = GameCollection
  132. def get_context_data(self, **kwargs):
  133. collection = self.get_object()
  134. order_by = self.request.GET.get("order_by", "release_date")
  135. object_list = collection.games.all()
  136. if order_by[0] == "-":
  137. order_by = order_by[1:]
  138. object_list = object_list.order_by(F(order_by).desc(nulls_last=True))
  139. else:
  140. object_list = object_list.order_by(F(order_by).asc(nulls_last=True))
  141. context = super(GameCollectionDetail, self).get_context_data(
  142. object_list=object_list, **kwargs
  143. )
  144. return context
  145. @login_required
  146. def trigger_rom_update(request):
  147. full_scan = request.GET.get("full_scan", False)
  148. if full_scan == "true":
  149. full_scan = True
  150. game_systems = request.GET.getlist("game_systems")
  151. if not game_systems:
  152. game_systems = list(settings.GAME_SYSTEM_DEFAULTS.keys())
  153. try:
  154. update_roms.delay(game_systems, full_scan=full_scan)
  155. except FileNotFoundError:
  156. return HttpResponse(
  157. json.dumps({"success": False, "msg": "Skyscraper is not installed"}),
  158. status=400,
  159. content_type="application/json",
  160. )
  161. return HttpResponse(
  162. json.dumps({"success": True, "msg": "Library scan started"}),
  163. status=200,
  164. content_type="application/json",
  165. )
  166. @login_required
  167. def library_update_status(request):
  168. update_task = TaskResult.objects.filter(
  169. task_name="games.tasks.update_roms",
  170. status__in=IN_PROGRESS_STATES,
  171. ).first()
  172. if not update_task:
  173. state = states.SUCCESS
  174. else:
  175. state = update_task.status
  176. return HttpResponse(
  177. json.dumps({"state": state}),
  178. status=200,
  179. content_type="application/json",
  180. )