views.py 52 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407
  1. from datetime import datetime, timedelta
  2. from celery.task.control import revoke
  3. from django.conf import settings
  4. from django.contrib import messages
  5. from django.contrib.auth.decorators import login_required
  6. from django.contrib.postgres.search import SearchQuery
  7. from django.core.mail import EmailMessage
  8. from django.db.models import Q
  9. from django.http import HttpResponseRedirect
  10. from django.shortcuts import get_object_or_404, render
  11. from django.template.defaultfilters import slugify
  12. from drf_yasg import openapi as openapi
  13. from drf_yasg.utils import swagger_auto_schema
  14. from rest_framework import permissions, status
  15. from rest_framework.exceptions import PermissionDenied
  16. from rest_framework.parsers import (
  17. FileUploadParser,
  18. FormParser,
  19. JSONParser,
  20. MultiPartParser,
  21. )
  22. from rest_framework.response import Response
  23. from rest_framework.settings import api_settings
  24. from rest_framework.views import APIView
  25. from actions.models import USER_MEDIA_ACTIONS, MediaAction
  26. from cms.custom_pagination import FastPaginationWithoutCount
  27. from cms.permissions import IsAuthorizedToAdd, IsUserOrEditor, user_allowed_to_upload
  28. from users.models import User
  29. from .forms import ContactForm, MediaForm, SubtitleForm
  30. from .helpers import clean_query, produce_ffmpeg_commands
  31. from .methods import (
  32. check_comment_for_mention,
  33. get_user_or_session,
  34. is_mediacms_editor,
  35. is_mediacms_manager,
  36. list_tasks,
  37. notify_user_on_comment,
  38. show_recommended_media,
  39. show_related_media,
  40. update_user_ratings,
  41. )
  42. from .models import (
  43. Category,
  44. Comment,
  45. EncodeProfile,
  46. Encoding,
  47. Media,
  48. Playlist,
  49. PlaylistMedia,
  50. Tag,
  51. )
  52. from .serializers import (
  53. CategorySerializer,
  54. CommentSerializer,
  55. EncodeProfileSerializer,
  56. MediaSearchSerializer,
  57. MediaSerializer,
  58. PlaylistDetailSerializer,
  59. PlaylistSerializer,
  60. SingleMediaSerializer,
  61. TagSerializer,
  62. )
  63. from .stop_words import STOP_WORDS
  64. from .tasks import save_user_action
  65. VALID_USER_ACTIONS = [action for action, name in USER_MEDIA_ACTIONS]
  66. def about(request):
  67. """About view"""
  68. context = {}
  69. return render(request, "cms/about.html", context)
  70. @login_required
  71. def add_subtitle(request):
  72. """Add subtitle view"""
  73. friendly_token = request.GET.get("m", "").strip()
  74. if not friendly_token:
  75. return HttpResponseRedirect("/")
  76. media = Media.objects.filter(friendly_token=friendly_token).first()
  77. if not media:
  78. return HttpResponseRedirect("/")
  79. if not (request.user == media.user or is_mediacms_editor(request.user) or is_mediacms_manager(request.user)):
  80. return HttpResponseRedirect("/")
  81. if request.method == "POST":
  82. form = SubtitleForm(media, request.POST, request.FILES)
  83. if form.is_valid():
  84. subtitle = form.save()
  85. messages.add_message(request, messages.INFO, "Subtitle was added!")
  86. return HttpResponseRedirect(subtitle.media.get_absolute_url())
  87. else:
  88. form = SubtitleForm(media_item=media)
  89. return render(request, "cms/add_subtitle.html", {"form": form})
  90. def categories(request):
  91. """List categories view"""
  92. context = {}
  93. return render(request, "cms/categories.html", context)
  94. def contact(request):
  95. """Contact view"""
  96. context = {}
  97. if request.method == "GET":
  98. form = ContactForm(request.user)
  99. context["form"] = form
  100. else:
  101. form = ContactForm(request.user, request.POST)
  102. if form.is_valid():
  103. if request.user.is_authenticated:
  104. from_email = request.user.email
  105. name = request.user.name
  106. else:
  107. from_email = request.POST.get("from_email")
  108. name = request.POST.get("name")
  109. message = request.POST.get("message")
  110. title = "[{}] - Contact form message received".format(settings.PORTAL_NAME)
  111. msg = """
  112. You have received a message through the contact form\n
  113. Sender name: %s
  114. Sender email: %s\n
  115. \n %s
  116. """ % (
  117. name,
  118. from_email,
  119. message,
  120. )
  121. email = EmailMessage(
  122. title,
  123. msg,
  124. settings.DEFAULT_FROM_EMAIL,
  125. settings.ADMIN_EMAIL_LIST,
  126. reply_to=[from_email],
  127. )
  128. email.send(fail_silently=True)
  129. success_msg = "Message was sent! Thanks for contacting"
  130. context["success_msg"] = success_msg
  131. return render(request, "cms/contact.html", context)
  132. def history(request):
  133. """Show personal history view"""
  134. context = {}
  135. return render(request, "cms/history.html", context)
  136. @login_required
  137. def edit_media(request):
  138. """Edit a media view"""
  139. friendly_token = request.GET.get("m", "").strip()
  140. if not friendly_token:
  141. return HttpResponseRedirect("/")
  142. media = Media.objects.filter(friendly_token=friendly_token).first()
  143. if not media:
  144. return HttpResponseRedirect("/")
  145. if not (request.user == media.user or is_mediacms_editor(request.user) or is_mediacms_manager(request.user)):
  146. return HttpResponseRedirect("/")
  147. if request.method == "POST":
  148. form = MediaForm(request.user, request.POST, request.FILES, instance=media)
  149. if form.is_valid():
  150. media = form.save()
  151. for tag in media.tags.all():
  152. media.tags.remove(tag)
  153. if form.cleaned_data.get("new_tags"):
  154. for tag in form.cleaned_data.get("new_tags").split(","):
  155. tag = slugify(tag)
  156. if tag:
  157. try:
  158. tag = Tag.objects.get(title=tag)
  159. except Tag.DoesNotExist:
  160. tag = Tag.objects.create(title=tag, user=request.user)
  161. if tag not in media.tags.all():
  162. media.tags.add(tag)
  163. messages.add_message(request, messages.INFO, "Media was edited!")
  164. return HttpResponseRedirect(media.get_absolute_url())
  165. else:
  166. form = MediaForm(request.user, instance=media)
  167. return render(
  168. request,
  169. "cms/edit_media.html",
  170. {"form": form, "add_subtitle_url": media.add_subtitle_url},
  171. )
  172. def embed_media(request):
  173. """Embed media view"""
  174. friendly_token = request.GET.get("m", "").strip()
  175. if not friendly_token:
  176. return HttpResponseRedirect("/")
  177. media = Media.objects.values("title").filter(friendly_token=friendly_token).first()
  178. if not media:
  179. return HttpResponseRedirect("/")
  180. context = {}
  181. context["media"] = friendly_token
  182. return render(request, "cms/embed.html", context)
  183. def featured_media(request):
  184. """List featured media view"""
  185. context = {}
  186. context["media"] = list(Media.objects.filter(Q(listable=True), featured=True))[:50]
  187. return render(request, "cms/featured-media.html", context)
  188. def index(request):
  189. """Index view"""
  190. context = {}
  191. context["media_featured"] = list(Media.objects.filter(Q(listable=True), featured=True))[:10]
  192. context["media_recommended"] = list(show_recommended_media(request, limit=10))
  193. context["media_latest"] = list(Media.objects.filter(Q(listable=True)).order_by("-add_date"))[:50]
  194. return render(request, "cms/index.html", context)
  195. def latest_media(request):
  196. """List latest media view"""
  197. context = {}
  198. context["media"] = list(Media.objects.filter(Q(listable=True)).order_by("-add_date"))[:50]
  199. return render(request, "cms/latest-media.html", context)
  200. def liked_media(request):
  201. """List user's liked media view"""
  202. context = {}
  203. return render(request, "cms/liked_media.html", context)
  204. @login_required
  205. def manage_users(request):
  206. """List users management view"""
  207. context = {}
  208. return render(request, "cms/manage_users.html", context)
  209. @login_required
  210. def manage_media(request):
  211. """List media management view"""
  212. context = {}
  213. return render(request, "cms/manage_media.html", context)
  214. @login_required
  215. def manage_comments(request):
  216. """List comments management view"""
  217. context = {}
  218. return render(request, "cms/manage_comments.html", context)
  219. def members(request):
  220. """List members view"""
  221. context = {}
  222. return render(request, "cms/members.html", context)
  223. def recommended_media(request):
  224. """List recommended media view"""
  225. context = {}
  226. context["media"] = list(show_recommended_media(request, limit=50))
  227. return render(request, "cms/recommended-media.html", context)
  228. def search(request):
  229. """Search view"""
  230. context = {}
  231. RSS_URL = f"/rss{request.environ['REQUEST_URI']}"
  232. context["RSS_URL"] = RSS_URL
  233. return render(request, "cms/search.html", context)
  234. def tags(request):
  235. """List tags view"""
  236. context = {}
  237. return render(request, "cms/tags.html", context)
  238. def tos(request):
  239. """Terms of service view"""
  240. context = {}
  241. return render(request, "cms/tos.html", context)
  242. def upload_media(request):
  243. """Upload media view"""
  244. from allauth.account.forms import LoginForm
  245. form = LoginForm()
  246. context = {}
  247. context["form"] = form
  248. context["can_add"] = user_allowed_to_upload(request)
  249. can_upload_exp = settings.CANNOT_ADD_MEDIA_MESSAGE
  250. context["can_upload_exp"] = can_upload_exp
  251. return render(request, "cms/add-media.html", context)
  252. def view_media(request):
  253. """View media view"""
  254. friendly_token = request.GET.get("m", "").strip()
  255. context = {}
  256. media = Media.objects.filter(friendly_token=friendly_token).first()
  257. if not media:
  258. context["media"] = None
  259. return render(request, "cms/media.html", context)
  260. user_or_session = get_user_or_session(request)
  261. save_user_action.delay(user_or_session, friendly_token=friendly_token, action="watch")
  262. context = {}
  263. context["media"] = friendly_token
  264. context["media_object"] = media
  265. context["CAN_DELETE_MEDIA"] = False
  266. context["CAN_EDIT_MEDIA"] = False
  267. context["CAN_DELETE_COMMENTS"] = False
  268. if request.user.is_authenticated:
  269. if (media.user.id == request.user.id) or is_mediacms_editor(request.user) or is_mediacms_manager(request.user):
  270. context["CAN_DELETE_MEDIA"] = True
  271. context["CAN_EDIT_MEDIA"] = True
  272. context["CAN_DELETE_COMMENTS"] = True
  273. return render(request, "cms/media.html", context)
  274. def view_playlist(request, friendly_token):
  275. """View playlist view"""
  276. try:
  277. playlist = Playlist.objects.get(friendly_token=friendly_token)
  278. except BaseException:
  279. playlist = None
  280. context = {}
  281. context["playlist"] = playlist
  282. context["media"] = [c.media for c in PlaylistMedia.objects.filter(playlist=playlist)]
  283. return render(request, "cms/playlist.html", context)
  284. class MediaList(APIView):
  285. """Media listings views"""
  286. permission_classes = (IsAuthorizedToAdd,)
  287. parser_classes = (MultiPartParser, FormParser, FileUploadParser)
  288. @swagger_auto_schema(
  289. manual_parameters=[
  290. openapi.Parameter(name='page', type=openapi.TYPE_INTEGER, in_=openapi.IN_QUERY, description='Page number'),
  291. openapi.Parameter(name='author', type=openapi.TYPE_STRING, in_=openapi.IN_QUERY, description='username'),
  292. openapi.Parameter(name='show', type=openapi.TYPE_STRING, in_=openapi.IN_QUERY, description='show', enum=['recommended', 'featured', 'latest']),
  293. ],
  294. tags=['Media'],
  295. operation_summary='List Media',
  296. operation_description='Lists all media',
  297. responses={200: MediaSerializer(many=True)},
  298. )
  299. def get(self, request, format=None):
  300. # Show media
  301. params = self.request.query_params
  302. show_param = params.get("show", "")
  303. author_param = params.get("author", "").strip()
  304. if author_param:
  305. user_queryset = User.objects.all()
  306. user = get_object_or_404(user_queryset, username=author_param)
  307. if show_param == "recommended":
  308. pagination_class = FastPaginationWithoutCount
  309. media = show_recommended_media(request, limit=50)
  310. else:
  311. pagination_class = api_settings.DEFAULT_PAGINATION_CLASS
  312. if author_param:
  313. # in case request.user is the user here, show
  314. # all media independant of state
  315. if self.request.user == user:
  316. basic_query = Q(user=user)
  317. else:
  318. basic_query = Q(listable=True, user=user)
  319. else:
  320. # base listings should show safe content
  321. basic_query = Q(listable=True)
  322. if show_param == "featured":
  323. media = Media.objects.filter(basic_query, featured=True)
  324. else:
  325. media = Media.objects.filter(basic_query).order_by("-add_date")
  326. paginator = pagination_class()
  327. if show_param != "recommended":
  328. media = media.prefetch_related("user")
  329. page = paginator.paginate_queryset(media, request)
  330. serializer = MediaSerializer(page, many=True, context={"request": request})
  331. return paginator.get_paginated_response(serializer.data)
  332. @swagger_auto_schema(
  333. manual_parameters=[
  334. openapi.Parameter(name="media_file", in_=openapi.IN_FORM, type=openapi.TYPE_FILE, required=True, description="media_file"),
  335. openapi.Parameter(name="description", in_=openapi.IN_FORM, type=openapi.TYPE_STRING, required=False, description="description"),
  336. openapi.Parameter(name="title", in_=openapi.IN_FORM, type=openapi.TYPE_STRING, required=False, description="title"),
  337. ],
  338. tags=['Media'],
  339. operation_summary='Add new Media',
  340. operation_description='Adds a new media, for authenticated users',
  341. responses={201: openapi.Response('response description', MediaSerializer), 401: 'bad request'},
  342. )
  343. def post(self, request, format=None):
  344. # Add new media
  345. serializer = MediaSerializer(data=request.data, context={"request": request})
  346. if serializer.is_valid():
  347. media_file = request.data["media_file"]
  348. serializer.save(user=request.user, media_file=media_file)
  349. return Response(serializer.data, status=status.HTTP_201_CREATED)
  350. return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
  351. class MediaDetail(APIView):
  352. """
  353. Retrieve, update or delete a media instance.
  354. """
  355. permission_classes = (permissions.IsAuthenticatedOrReadOnly, IsUserOrEditor)
  356. parser_classes = (MultiPartParser, FormParser, FileUploadParser)
  357. def get_object(self, friendly_token, password=None):
  358. try:
  359. media = Media.objects.select_related("user").prefetch_related("encodings__profile").get(friendly_token=friendly_token)
  360. # this need be explicitly called, and will call
  361. # has_object_permission() after has_permission has succeeded
  362. self.check_object_permissions(self.request, media)
  363. if media.state == "private" and not (self.request.user == media.user or is_mediacms_editor(self.request.user)):
  364. if (not password) or (not media.password) or (password != media.password):
  365. return Response(
  366. {"detail": "media is private"},
  367. status=status.HTTP_401_UNAUTHORIZED,
  368. )
  369. return media
  370. except PermissionDenied:
  371. return Response({"detail": "bad permissions"}, status=status.HTTP_401_UNAUTHORIZED)
  372. except BaseException:
  373. return Response(
  374. {"detail": "media file does not exist"},
  375. status=status.HTTP_400_BAD_REQUEST,
  376. )
  377. @swagger_auto_schema(
  378. manual_parameters=[
  379. openapi.Parameter(name='friendly_token', type=openapi.TYPE_STRING, in_=openapi.IN_PATH, description='unique identifier', required=True),
  380. ],
  381. tags=['Media'],
  382. operation_summary='Get information for Media',
  383. operation_description='Get information for a media',
  384. responses={200: SingleMediaSerializer(), 400: 'bad request'},
  385. )
  386. def get(self, request, friendly_token, format=None):
  387. # Get media details
  388. password = request.GET.get("password")
  389. media = self.get_object(friendly_token, password=password)
  390. if isinstance(media, Response):
  391. return media
  392. serializer = SingleMediaSerializer(media, context={"request": request})
  393. if media.state == "private":
  394. related_media = []
  395. else:
  396. related_media = show_related_media(media, request=request, limit=100)
  397. related_media_serializer = MediaSerializer(related_media, many=True, context={"request": request})
  398. related_media = related_media_serializer.data
  399. ret = serializer.data
  400. # update rattings info with user specific ratings
  401. # eg user has already rated for this media
  402. # this only affects user rating and only if enabled
  403. if settings.ALLOW_RATINGS and ret.get("ratings_info") and not request.user.is_anonymous:
  404. ret["ratings_info"] = update_user_ratings(request.user, media, ret.get("ratings_info"))
  405. ret["related_media"] = related_media
  406. return Response(ret)
  407. @swagger_auto_schema(
  408. manual_parameters=[
  409. openapi.Parameter(name='friendly_token', type=openapi.TYPE_STRING, in_=openapi.IN_PATH, description='unique identifier', required=True),
  410. openapi.Parameter(name='type', type=openapi.TYPE_STRING, in_=openapi.IN_FORM, description='action to perform', enum=['encode', 'review']),
  411. openapi.Parameter(
  412. name='encoding_profiles',
  413. type=openapi.TYPE_ARRAY,
  414. items=openapi.Items(type=openapi.TYPE_STRING),
  415. in_=openapi.IN_FORM,
  416. description='if action to perform is encode, need to specify list of ids of encoding profiles',
  417. ),
  418. openapi.Parameter(name='result', type=openapi.TYPE_BOOLEAN, in_=openapi.IN_FORM, description='if action is review, this is the result (True for reviewed, False for not reviewed)'),
  419. ],
  420. tags=['Media'],
  421. operation_summary='Run action on Media',
  422. operation_description='Actions for a media, for MediaCMS editors and managers',
  423. responses={201: 'action created', 400: 'bad request'},
  424. operation_id='media_manager_actions',
  425. )
  426. def post(self, request, friendly_token, format=None):
  427. """superuser actions
  428. Available only to MediaCMS editors and managers
  429. Action is a POST variable, review and encode are implemented
  430. """
  431. media = self.get_object(friendly_token)
  432. if isinstance(media, Response):
  433. return media
  434. if not (is_mediacms_editor(request.user) or is_mediacms_manager(request.user)):
  435. return Response({"detail": "not allowed"}, status=status.HTTP_400_BAD_REQUEST)
  436. action = request.data.get("type")
  437. profiles_list = request.data.get("encoding_profiles")
  438. result = request.data.get("result", True)
  439. if action == "encode":
  440. # Create encoding tasks for specific profiles
  441. valid_profiles = []
  442. if profiles_list:
  443. if isinstance(profiles_list, list):
  444. for p in profiles_list:
  445. p = EncodeProfile.objects.filter(id=p).first()
  446. if p:
  447. valid_profiles.append(p)
  448. elif isinstance(profiles_list, str):
  449. try:
  450. p = EncodeProfile.objects.filter(id=int(profiles_list)).first()
  451. valid_profiles.append(p)
  452. except ValueError:
  453. return Response(
  454. {"detail": "encoding_profiles must be int or list of ints of valid encode profiles"},
  455. status=status.HTTP_400_BAD_REQUEST,
  456. )
  457. media.encode(profiles=valid_profiles)
  458. return Response({"detail": "media will be encoded"}, status=status.HTTP_201_CREATED)
  459. elif action == "review":
  460. if result:
  461. media.is_reviewed = True
  462. elif result is False:
  463. media.is_reviewed = False
  464. media.save(update_fields=["is_reviewed"])
  465. return Response({"detail": "media reviewed set"}, status=status.HTTP_201_CREATED)
  466. return Response(
  467. {"detail": "not valid action or no action specified"},
  468. status=status.HTTP_400_BAD_REQUEST,
  469. )
  470. @swagger_auto_schema(
  471. manual_parameters=[
  472. openapi.Parameter(name="description", in_=openapi.IN_FORM, type=openapi.TYPE_STRING, required=False, description="description"),
  473. openapi.Parameter(name="title", in_=openapi.IN_FORM, type=openapi.TYPE_STRING, required=False, description="title"),
  474. openapi.Parameter(name="media_file", in_=openapi.IN_FORM, type=openapi.TYPE_FILE, required=False, description="media_file"),
  475. ],
  476. tags=['Media'],
  477. operation_summary='Update Media',
  478. operation_description='Update a Media, for Media uploader',
  479. responses={201: openapi.Response('response description', MediaSerializer), 401: 'bad request'},
  480. )
  481. def put(self, request, friendly_token, format=None):
  482. # Update a media object
  483. media = self.get_object(friendly_token)
  484. if isinstance(media, Response):
  485. return media
  486. serializer = MediaSerializer(media, data=request.data, context={"request": request})
  487. if serializer.is_valid():
  488. if request.data.get('media_file'):
  489. media_file = request.data["media_file"]
  490. serializer.save(user=request.user, media_file=media_file)
  491. else:
  492. serializer.save(user=request.user)
  493. return Response(serializer.data, status=status.HTTP_201_CREATED)
  494. return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
  495. @swagger_auto_schema(
  496. manual_parameters=[
  497. openapi.Parameter(name='friendly_token', type=openapi.TYPE_STRING, in_=openapi.IN_PATH, description='unique identifier', required=True),
  498. ],
  499. tags=['Media'],
  500. operation_summary='Delete Media',
  501. operation_description='Delete a Media, for MediaCMS editors and managers',
  502. responses={
  503. 204: 'no content',
  504. },
  505. )
  506. def delete(self, request, friendly_token, format=None):
  507. # Delete a media object
  508. media = self.get_object(friendly_token)
  509. if isinstance(media, Response):
  510. return media
  511. media.delete()
  512. return Response(status=status.HTTP_204_NO_CONTENT)
  513. class MediaActions(APIView):
  514. """
  515. Retrieve, update or delete a media action instance.
  516. """
  517. permission_classes = (permissions.AllowAny,)
  518. parser_classes = (JSONParser,)
  519. def get_object(self, friendly_token):
  520. try:
  521. media = Media.objects.select_related("user").prefetch_related("encodings__profile").get(friendly_token=friendly_token)
  522. if media.state == "private" and self.request.user != media.user:
  523. return Response({"detail": "media is private"}, status=status.HTTP_400_BAD_REQUEST)
  524. return media
  525. except PermissionDenied:
  526. return Response({"detail": "bad permissions"}, status=status.HTTP_400_BAD_REQUEST)
  527. except BaseException:
  528. return Response(
  529. {"detail": "media file does not exist"},
  530. status=status.HTTP_400_BAD_REQUEST,
  531. )
  532. @swagger_auto_schema(
  533. manual_parameters=[],
  534. tags=['Media'],
  535. operation_summary='to_be_written',
  536. operation_description='to_be_written',
  537. )
  538. def get(self, request, friendly_token, format=None):
  539. # show date and reason for each time media was reported
  540. media = self.get_object(friendly_token)
  541. if isinstance(media, Response):
  542. return media
  543. ret = {}
  544. reported = MediaAction.objects.filter(media=media, action="report")
  545. ret["reported"] = []
  546. for rep in reported:
  547. item = {"reported_date": rep.action_date, "reason": rep.extra_info}
  548. ret["reported"].append(item)
  549. return Response(ret, status=status.HTTP_200_OK)
  550. @swagger_auto_schema(
  551. manual_parameters=[],
  552. tags=['Media'],
  553. operation_summary='to_be_written',
  554. operation_description='to_be_written',
  555. )
  556. def post(self, request, friendly_token, format=None):
  557. # perform like/dislike/report actions
  558. media = self.get_object(friendly_token)
  559. if isinstance(media, Response):
  560. return media
  561. action = request.data.get("type")
  562. extra = request.data.get("extra_info")
  563. if request.user.is_anonymous:
  564. # there is a list of allowed actions for
  565. # anonymous users, specified in settings
  566. if action not in settings.ALLOW_ANONYMOUS_ACTIONS:
  567. return Response(
  568. {"detail": "action allowed on logged in users only"},
  569. status=status.HTTP_400_BAD_REQUEST,
  570. )
  571. if action:
  572. user_or_session = get_user_or_session(request)
  573. save_user_action.delay(
  574. user_or_session,
  575. friendly_token=media.friendly_token,
  576. action=action,
  577. extra_info=extra,
  578. )
  579. return Response({"detail": "action received"}, status=status.HTTP_201_CREATED)
  580. else:
  581. return Response({"detail": "no action specified"}, status=status.HTTP_400_BAD_REQUEST)
  582. @swagger_auto_schema(
  583. manual_parameters=[],
  584. tags=['Media'],
  585. operation_summary='to_be_written',
  586. operation_description='to_be_written',
  587. )
  588. def delete(self, request, friendly_token, format=None):
  589. media = self.get_object(friendly_token)
  590. if isinstance(media, Response):
  591. return media
  592. if not request.user.is_superuser:
  593. return Response({"detail": "not allowed"}, status=status.HTTP_400_BAD_REQUEST)
  594. action = request.data.get("type")
  595. if action:
  596. if action == "report": # delete reported actions
  597. MediaAction.objects.filter(media=media, action="report").delete()
  598. media.reported_times = 0
  599. media.save(update_fields=["reported_times"])
  600. return Response(
  601. {"detail": "reset reported times counter"},
  602. status=status.HTTP_201_CREATED,
  603. )
  604. else:
  605. return Response({"detail": "no action specified"}, status=status.HTTP_400_BAD_REQUEST)
  606. class MediaSearch(APIView):
  607. """
  608. Retrieve results for searc
  609. Only GET is implemented here
  610. """
  611. parser_classes = (JSONParser,)
  612. @swagger_auto_schema(
  613. manual_parameters=[],
  614. tags=['Search'],
  615. operation_summary='to_be_written',
  616. operation_description='to_be_written',
  617. )
  618. def get(self, request, format=None):
  619. params = self.request.query_params
  620. query = params.get("q", "").strip().lower()
  621. category = params.get("c", "").strip()
  622. tag = params.get("t", "").strip()
  623. ordering = params.get("ordering", "").strip()
  624. sort_by = params.get("sort_by", "").strip()
  625. media_type = params.get("media_type", "").strip()
  626. author = params.get("author", "").strip()
  627. upload_date = params.get('upload_date', '').strip()
  628. sort_by_options = ["title", "add_date", "edit_date", "views", "likes"]
  629. if sort_by not in sort_by_options:
  630. sort_by = "add_date"
  631. if ordering == "asc":
  632. ordering = ""
  633. else:
  634. ordering = "-"
  635. if media_type not in ["video", "image", "audio", "pdf"]:
  636. media_type = None
  637. if not (query or category or tag):
  638. ret = {}
  639. return Response(ret, status=status.HTTP_200_OK)
  640. media = Media.objects.filter(state="public", is_reviewed=True)
  641. if query:
  642. # move this processing to a prepare_query function
  643. query = clean_query(query)
  644. q_parts = [q_part.rstrip("y") for q_part in query.split() if q_part not in STOP_WORDS]
  645. if q_parts:
  646. query = SearchQuery(q_parts[0] + ":*", search_type="raw")
  647. for part in q_parts[1:]:
  648. query &= SearchQuery(part + ":*", search_type="raw")
  649. else:
  650. query = None
  651. if query:
  652. media = media.filter(search=query)
  653. if tag:
  654. media = media.filter(tags__title=tag)
  655. if category:
  656. media = media.filter(category__title__contains=category)
  657. if media_type:
  658. media = media.filter(media_type=media_type)
  659. if author:
  660. media = media.filter(user__username=author)
  661. if upload_date:
  662. gte = None
  663. if upload_date == 'today':
  664. gte = datetime.now().date()
  665. if upload_date == 'this_week':
  666. gte = datetime.now() - timedelta(days=7)
  667. if upload_date == 'this_month':
  668. year = datetime.now().date().year
  669. month = datetime.now().date().month
  670. gte = datetime(year, month, 1)
  671. if upload_date == 'this_year':
  672. year = datetime.now().date().year
  673. gte = datetime(year, 1, 1)
  674. if gte:
  675. media = media.filter(add_date__gte=gte)
  676. media = media.order_by(f"{ordering}{sort_by}")
  677. if self.request.query_params.get("show", "").strip() == "titles":
  678. media = media.values("title")[:40]
  679. return Response(media, status=status.HTTP_200_OK)
  680. else:
  681. media = media.prefetch_related("user")
  682. if category or tag:
  683. pagination_class = api_settings.DEFAULT_PAGINATION_CLASS
  684. else:
  685. # pagination_class = FastPaginationWithoutCount
  686. pagination_class = api_settings.DEFAULT_PAGINATION_CLASS
  687. paginator = pagination_class()
  688. page = paginator.paginate_queryset(media, request)
  689. serializer = MediaSearchSerializer(page, many=True, context={"request": request})
  690. return paginator.get_paginated_response(serializer.data)
  691. class PlaylistList(APIView):
  692. """Playlists listings and creation views"""
  693. permission_classes = (permissions.IsAuthenticatedOrReadOnly, IsAuthorizedToAdd)
  694. parser_classes = (JSONParser, MultiPartParser, FormParser, FileUploadParser)
  695. @swagger_auto_schema(
  696. manual_parameters=[],
  697. tags=['Playlists'],
  698. operation_summary='to_be_written',
  699. operation_description='to_be_written',
  700. responses={
  701. 200: openapi.Response('response description', PlaylistSerializer(many=True)),
  702. },
  703. )
  704. def get(self, request, format=None):
  705. pagination_class = api_settings.DEFAULT_PAGINATION_CLASS
  706. paginator = pagination_class()
  707. playlists = Playlist.objects.filter().prefetch_related("user")
  708. if "author" in self.request.query_params:
  709. author = self.request.query_params["author"].strip()
  710. playlists = playlists.filter(user__username=author)
  711. page = paginator.paginate_queryset(playlists, request)
  712. serializer = PlaylistSerializer(page, many=True, context={"request": request})
  713. return paginator.get_paginated_response(serializer.data)
  714. @swagger_auto_schema(
  715. manual_parameters=[],
  716. tags=['Playlists'],
  717. operation_summary='to_be_written',
  718. operation_description='to_be_written',
  719. )
  720. def post(self, request, format=None):
  721. serializer = PlaylistSerializer(data=request.data, context={"request": request})
  722. if serializer.is_valid():
  723. serializer.save(user=request.user)
  724. return Response(serializer.data, status=status.HTTP_201_CREATED)
  725. return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
  726. class PlaylistDetail(APIView):
  727. """Playlist related views"""
  728. permission_classes = (permissions.IsAuthenticatedOrReadOnly, IsUserOrEditor)
  729. parser_classes = (JSONParser, MultiPartParser, FormParser, FileUploadParser)
  730. def get_playlist(self, friendly_token):
  731. try:
  732. playlist = Playlist.objects.get(friendly_token=friendly_token)
  733. self.check_object_permissions(self.request, playlist)
  734. return playlist
  735. except PermissionDenied:
  736. return Response({"detail": "not enough permissions"}, status=status.HTTP_400_BAD_REQUEST)
  737. except BaseException:
  738. return Response(
  739. {"detail": "Playlist does not exist"},
  740. status=status.HTTP_400_BAD_REQUEST,
  741. )
  742. @swagger_auto_schema(
  743. manual_parameters=[],
  744. tags=['Playlists'],
  745. operation_summary='to_be_written',
  746. operation_description='to_be_written',
  747. )
  748. def get(self, request, friendly_token, format=None):
  749. playlist = self.get_playlist(friendly_token)
  750. if isinstance(playlist, Response):
  751. return playlist
  752. serializer = PlaylistDetailSerializer(playlist, context={"request": request})
  753. playlist_media = PlaylistMedia.objects.filter(playlist=playlist).prefetch_related("media__user")
  754. playlist_media = [c.media for c in playlist_media]
  755. playlist_media_serializer = MediaSerializer(playlist_media, many=True, context={"request": request})
  756. ret = serializer.data
  757. ret["playlist_media"] = playlist_media_serializer.data
  758. return Response(ret)
  759. @swagger_auto_schema(
  760. manual_parameters=[],
  761. tags=['Playlists'],
  762. operation_summary='to_be_written',
  763. operation_description='to_be_written',
  764. )
  765. def post(self, request, friendly_token, format=None):
  766. playlist = self.get_playlist(friendly_token)
  767. if isinstance(playlist, Response):
  768. return playlist
  769. serializer = PlaylistDetailSerializer(playlist, data=request.data, context={"request": request})
  770. if serializer.is_valid():
  771. serializer.save(user=request.user)
  772. return Response(serializer.data, status=status.HTTP_201_CREATED)
  773. return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
  774. @swagger_auto_schema(
  775. manual_parameters=[],
  776. tags=['Playlists'],
  777. operation_summary='to_be_written',
  778. operation_description='to_be_written',
  779. )
  780. def put(self, request, friendly_token, format=None):
  781. playlist = self.get_playlist(friendly_token)
  782. if isinstance(playlist, Response):
  783. return playlist
  784. action = request.data.get("type")
  785. media_friendly_token = request.data.get("media_friendly_token")
  786. ordering = 0
  787. if request.data.get("ordering"):
  788. try:
  789. ordering = int(request.data.get("ordering"))
  790. except ValueError:
  791. pass
  792. if action in ["add", "remove", "ordering"]:
  793. media = Media.objects.filter(friendly_token=media_friendly_token).first()
  794. if media:
  795. if action == "add":
  796. media_in_playlist = PlaylistMedia.objects.filter(playlist=playlist).count()
  797. if media_in_playlist >= settings.MAX_MEDIA_PER_PLAYLIST:
  798. return Response(
  799. {"detail": "max number of media for a Playlist reached"},
  800. status=status.HTTP_400_BAD_REQUEST,
  801. )
  802. else:
  803. obj, created = PlaylistMedia.objects.get_or_create(
  804. playlist=playlist,
  805. media=media,
  806. ordering=media_in_playlist + 1,
  807. )
  808. obj.save()
  809. return Response(
  810. {"detail": "media added to Playlist"},
  811. status=status.HTTP_201_CREATED,
  812. )
  813. elif action == "remove":
  814. PlaylistMedia.objects.filter(playlist=playlist, media=media).delete()
  815. return Response(
  816. {"detail": "media removed from Playlist"},
  817. status=status.HTTP_201_CREATED,
  818. )
  819. elif action == "ordering":
  820. if ordering:
  821. playlist.set_ordering(media, ordering)
  822. return Response(
  823. {"detail": "new ordering set"},
  824. status=status.HTTP_201_CREATED,
  825. )
  826. else:
  827. return Response({"detail": "media is not valid"}, status=status.HTTP_400_BAD_REQUEST)
  828. return Response(
  829. {"detail": "invalid or not specified action"},
  830. status=status.HTTP_400_BAD_REQUEST,
  831. )
  832. @swagger_auto_schema(
  833. manual_parameters=[],
  834. tags=['Playlists'],
  835. operation_summary='to_be_written',
  836. operation_description='to_be_written',
  837. )
  838. def delete(self, request, friendly_token, format=None):
  839. playlist = self.get_playlist(friendly_token)
  840. if isinstance(playlist, Response):
  841. return playlist
  842. playlist.delete()
  843. return Response(status=status.HTTP_204_NO_CONTENT)
  844. class EncodingDetail(APIView):
  845. """Experimental. This View is used by remote workers
  846. Needs heavy testing and documentation.
  847. """
  848. permission_classes = (permissions.IsAdminUser,)
  849. parser_classes = (JSONParser, MultiPartParser, FormParser, FileUploadParser)
  850. @swagger_auto_schema(auto_schema=None)
  851. def post(self, request, encoding_id):
  852. ret = {}
  853. force = request.data.get("force", False)
  854. task_id = request.data.get("task_id", False)
  855. action = request.data.get("action", "")
  856. chunk = request.data.get("chunk", False)
  857. chunk_file_path = request.data.get("chunk_file_path", "")
  858. encoding_status = request.data.get("status", "")
  859. progress = request.data.get("progress", "")
  860. commands = request.data.get("commands", "")
  861. logs = request.data.get("logs", "")
  862. retries = request.data.get("retries", "")
  863. worker = request.data.get("worker", "")
  864. temp_file = request.data.get("temp_file", "")
  865. total_run_time = request.data.get("total_run_time", "")
  866. if action == "start":
  867. try:
  868. encoding = Encoding.objects.get(id=encoding_id)
  869. media = encoding.media
  870. profile = encoding.profile
  871. except BaseException:
  872. Encoding.objects.filter(id=encoding_id).delete()
  873. return Response({"status": "fail"}, status=status.HTTP_400_BAD_REQUEST)
  874. # TODO: break chunk True/False logic here
  875. if (
  876. Encoding.objects.filter(
  877. media=media,
  878. profile=profile,
  879. chunk=chunk,
  880. chunk_file_path=chunk_file_path,
  881. ).count()
  882. > 1 # noqa
  883. and force is False # noqa
  884. ):
  885. Encoding.objects.filter(id=encoding_id).delete()
  886. return Response({"status": "fail"}, status=status.HTTP_400_BAD_REQUEST)
  887. else:
  888. Encoding.objects.filter(
  889. media=media,
  890. profile=profile,
  891. chunk=chunk,
  892. chunk_file_path=chunk_file_path,
  893. ).exclude(id=encoding.id).delete()
  894. encoding.status = "running"
  895. if task_id:
  896. encoding.task_id = task_id
  897. encoding.save()
  898. if chunk:
  899. original_media_path = chunk_file_path
  900. original_media_md5sum = encoding.md5sum
  901. original_media_url = settings.SSL_FRONTEND_HOST + encoding.media_chunk_url
  902. else:
  903. original_media_path = media.media_file.path
  904. original_media_md5sum = media.md5sum
  905. original_media_url = settings.SSL_FRONTEND_HOST + media.original_media_url
  906. ret["original_media_url"] = original_media_url
  907. ret["original_media_path"] = original_media_path
  908. ret["original_media_md5sum"] = original_media_md5sum
  909. # generating the commands here, and will replace these with temporary
  910. # files created on the remote server
  911. tf = "TEMP_FILE_REPLACE"
  912. tfpass = "TEMP_FPASS_FILE_REPLACE"
  913. ffmpeg_commands = produce_ffmpeg_commands(
  914. original_media_path,
  915. media.media_info,
  916. resolution=profile.resolution,
  917. codec=profile.codec,
  918. output_filename=tf,
  919. pass_file=tfpass,
  920. chunk=chunk,
  921. )
  922. if not ffmpeg_commands:
  923. encoding.delete()
  924. return Response({"status": "fail"}, status=status.HTTP_400_BAD_REQUEST)
  925. ret["duration"] = media.duration
  926. ret["ffmpeg_commands"] = ffmpeg_commands
  927. ret["profile_extension"] = profile.extension
  928. return Response(ret, status=status.HTTP_201_CREATED)
  929. elif action == "update_fields":
  930. try:
  931. encoding = Encoding.objects.get(id=encoding_id)
  932. except BaseException:
  933. return Response({"status": "fail"}, status=status.HTTP_400_BAD_REQUEST)
  934. to_update = ["size", "update_date"]
  935. if encoding_status:
  936. encoding.status = encoding_status
  937. to_update.append("status")
  938. if progress:
  939. encoding.progress = progress
  940. to_update.append("progress")
  941. if logs:
  942. encoding.logs = logs
  943. to_update.append("logs")
  944. if commands:
  945. encoding.commands = commands
  946. to_update.append("commands")
  947. if task_id:
  948. encoding.task_id = task_id
  949. to_update.append("task_id")
  950. if total_run_time:
  951. encoding.total_run_time = total_run_time
  952. to_update.append("total_run_time")
  953. if worker:
  954. encoding.worker = worker
  955. to_update.append("worker")
  956. if temp_file:
  957. encoding.temp_file = temp_file
  958. to_update.append("temp_file")
  959. if retries:
  960. encoding.retries = retries
  961. to_update.append("retries")
  962. try:
  963. encoding.save(update_fields=to_update)
  964. except BaseException:
  965. return Response({"status": "fail"}, status=status.HTTP_400_BAD_REQUEST)
  966. return Response({"status": "success"}, status=status.HTTP_201_CREATED)
  967. @swagger_auto_schema(auto_schema=None)
  968. def put(self, request, encoding_id, format=None):
  969. encoding_file = request.data["file"]
  970. encoding = Encoding.objects.filter(id=encoding_id).first()
  971. if not encoding:
  972. return Response(
  973. {"detail": "encoding does not exist"},
  974. status=status.HTTP_400_BAD_REQUEST,
  975. )
  976. encoding.media_file = encoding_file
  977. encoding.save()
  978. return Response({"detail": "ok"}, status=status.HTTP_201_CREATED)
  979. class CommentList(APIView):
  980. permission_classes = (permissions.IsAuthenticatedOrReadOnly, IsAuthorizedToAdd)
  981. parser_classes = (JSONParser, MultiPartParser, FormParser, FileUploadParser)
  982. @swagger_auto_schema(
  983. manual_parameters=[
  984. openapi.Parameter(name='page', type=openapi.TYPE_INTEGER, in_=openapi.IN_QUERY, description='Page number'),
  985. openapi.Parameter(name='author', type=openapi.TYPE_STRING, in_=openapi.IN_QUERY, description='username'),
  986. ],
  987. tags=['Comments'],
  988. operation_summary='Lists Comments',
  989. operation_description='Paginated listing of all comments',
  990. responses={
  991. 200: openapi.Response('response description', CommentSerializer(many=True)),
  992. },
  993. )
  994. def get(self, request, format=None):
  995. pagination_class = api_settings.DEFAULT_PAGINATION_CLASS
  996. paginator = pagination_class()
  997. comments = Comment.objects.filter()
  998. comments = comments.prefetch_related("user")
  999. comments = comments.prefetch_related("media")
  1000. params = self.request.query_params
  1001. if "author" in params:
  1002. author_param = params["author"].strip()
  1003. user_queryset = User.objects.all()
  1004. user = get_object_or_404(user_queryset, username=author_param)
  1005. comments = comments.filter(user=user)
  1006. page = paginator.paginate_queryset(comments, request)
  1007. serializer = CommentSerializer(page, many=True, context={"request": request})
  1008. return paginator.get_paginated_response(serializer.data)
  1009. class CommentDetail(APIView):
  1010. """Comments related views
  1011. Listings of comments for a media (GET)
  1012. Create comment (POST)
  1013. Delete comment (DELETE)
  1014. """
  1015. permission_classes = (IsAuthorizedToAdd,)
  1016. parser_classes = (JSONParser, MultiPartParser, FormParser, FileUploadParser)
  1017. def get_object(self, friendly_token):
  1018. try:
  1019. media = Media.objects.select_related("user").get(friendly_token=friendly_token)
  1020. self.check_object_permissions(self.request, media)
  1021. if media.state == "private" and self.request.user != media.user:
  1022. return Response({"detail": "media is private"}, status=status.HTTP_400_BAD_REQUEST)
  1023. return media
  1024. except PermissionDenied:
  1025. return Response({"detail": "bad permissions"}, status=status.HTTP_400_BAD_REQUEST)
  1026. except BaseException:
  1027. return Response(
  1028. {"detail": "media file does not exist"},
  1029. status=status.HTTP_400_BAD_REQUEST,
  1030. )
  1031. @swagger_auto_schema(
  1032. manual_parameters=[],
  1033. tags=['Media'],
  1034. operation_summary='to_be_written',
  1035. operation_description='to_be_written',
  1036. )
  1037. def get(self, request, friendly_token):
  1038. # list comments for a media
  1039. media = self.get_object(friendly_token)
  1040. if isinstance(media, Response):
  1041. return media
  1042. comments = media.comments.filter().prefetch_related("user")
  1043. pagination_class = api_settings.DEFAULT_PAGINATION_CLASS
  1044. paginator = pagination_class()
  1045. page = paginator.paginate_queryset(comments, request)
  1046. serializer = CommentSerializer(page, many=True, context={"request": request})
  1047. return paginator.get_paginated_response(serializer.data)
  1048. @swagger_auto_schema(
  1049. manual_parameters=[],
  1050. tags=['Media'],
  1051. operation_summary='to_be_written',
  1052. operation_description='to_be_written',
  1053. )
  1054. def delete(self, request, friendly_token, uid=None):
  1055. """Delete a comment
  1056. Administrators, MediaCMS editors and managers,
  1057. media owner, and comment owners, can delete a comment
  1058. """
  1059. if uid:
  1060. try:
  1061. comment = Comment.objects.get(uid=uid)
  1062. except BaseException:
  1063. return Response(
  1064. {"detail": "comment does not exist"},
  1065. status=status.HTTP_400_BAD_REQUEST,
  1066. )
  1067. if (comment.user == self.request.user) or comment.media.user == self.request.user or is_mediacms_editor(self.request.user):
  1068. comment.delete()
  1069. else:
  1070. return Response({"detail": "bad permissions"}, status=status.HTTP_400_BAD_REQUEST)
  1071. return Response(status=status.HTTP_204_NO_CONTENT)
  1072. @swagger_auto_schema(
  1073. manual_parameters=[],
  1074. tags=['Media'],
  1075. operation_summary='to_be_written',
  1076. operation_description='to_be_written',
  1077. )
  1078. def post(self, request, friendly_token):
  1079. """Create a comment"""
  1080. media = self.get_object(friendly_token)
  1081. if isinstance(media, Response):
  1082. return media
  1083. if not media.enable_comments:
  1084. return Response(
  1085. {"detail": "comments not allowed here"},
  1086. status=status.HTTP_400_BAD_REQUEST,
  1087. )
  1088. serializer = CommentSerializer(data=request.data, context={"request": request})
  1089. if serializer.is_valid():
  1090. serializer.save(user=request.user, media=media)
  1091. if request.user != media.user:
  1092. notify_user_on_comment(friendly_token=media.friendly_token)
  1093. # here forward the comment to check if a user was mentioned
  1094. if settings.ALLOW_MENTION_IN_COMMENTS:
  1095. check_comment_for_mention(friendly_token=media.friendly_token, comment_text=serializer.data['text'])
  1096. return Response(serializer.data, status=status.HTTP_201_CREATED)
  1097. return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)
  1098. class UserActions(APIView):
  1099. parser_classes = (JSONParser,)
  1100. @swagger_auto_schema(
  1101. manual_parameters=[
  1102. openapi.Parameter(name='action', type=openapi.TYPE_STRING, in_=openapi.IN_PATH, description='action', required=True, enum=VALID_USER_ACTIONS),
  1103. ],
  1104. tags=['Users'],
  1105. operation_summary='List user actions',
  1106. operation_description='Lists user actions',
  1107. )
  1108. def get(self, request, action):
  1109. media = []
  1110. if action in VALID_USER_ACTIONS:
  1111. if request.user.is_authenticated:
  1112. media = Media.objects.select_related("user").filter(mediaactions__user=request.user, mediaactions__action=action).order_by("-mediaactions__action_date")
  1113. elif request.session.session_key:
  1114. media = (
  1115. Media.objects.select_related("user")
  1116. .filter(
  1117. mediaactions__session_key=request.session.session_key,
  1118. mediaactions__action=action,
  1119. )
  1120. .order_by("-mediaactions__action_date")
  1121. )
  1122. pagination_class = api_settings.DEFAULT_PAGINATION_CLASS
  1123. paginator = pagination_class()
  1124. page = paginator.paginate_queryset(media, request)
  1125. serializer = MediaSerializer(page, many=True, context={"request": request})
  1126. return paginator.get_paginated_response(serializer.data)
  1127. class CategoryList(APIView):
  1128. """List categories"""
  1129. @swagger_auto_schema(
  1130. manual_parameters=[],
  1131. tags=['Categories'],
  1132. operation_summary='Lists Categories',
  1133. operation_description='Lists all categories',
  1134. responses={
  1135. 200: openapi.Response('response description', CategorySerializer),
  1136. },
  1137. )
  1138. def get(self, request, format=None):
  1139. categories = Category.objects.filter().order_by("title")
  1140. serializer = CategorySerializer(categories, many=True, context={"request": request})
  1141. ret = serializer.data
  1142. return Response(ret)
  1143. class TagList(APIView):
  1144. """List tags"""
  1145. @swagger_auto_schema(
  1146. manual_parameters=[
  1147. openapi.Parameter(name='page', type=openapi.TYPE_INTEGER, in_=openapi.IN_QUERY, description='Page number'),
  1148. ],
  1149. tags=['Tags'],
  1150. operation_summary='Lists Tags',
  1151. operation_description='Paginated listing of all tags',
  1152. responses={
  1153. 200: openapi.Response('response description', TagSerializer),
  1154. },
  1155. )
  1156. def get(self, request, format=None):
  1157. tags = Tag.objects.filter().order_by("-media_count")
  1158. pagination_class = api_settings.DEFAULT_PAGINATION_CLASS
  1159. paginator = pagination_class()
  1160. page = paginator.paginate_queryset(tags, request)
  1161. serializer = TagSerializer(page, many=True, context={"request": request})
  1162. return paginator.get_paginated_response(serializer.data)
  1163. class EncodeProfileList(APIView):
  1164. """List encode profiles"""
  1165. @swagger_auto_schema(
  1166. manual_parameters=[],
  1167. tags=['Encoding Profiles'],
  1168. operation_summary='List Encoding Profiles',
  1169. operation_description='Lists all encoding profiles for videos',
  1170. responses={200: EncodeProfileSerializer(many=True)},
  1171. )
  1172. def get(self, request, format=None):
  1173. profiles = EncodeProfile.objects.all()
  1174. serializer = EncodeProfileSerializer(profiles, many=True, context={"request": request})
  1175. return Response(serializer.data)
  1176. class TasksList(APIView):
  1177. """List tasks"""
  1178. swagger_schema = None
  1179. permission_classes = (permissions.IsAdminUser,)
  1180. def get(self, request, format=None):
  1181. ret = list_tasks()
  1182. return Response(ret)
  1183. class TaskDetail(APIView):
  1184. """Cancel a task"""
  1185. swagger_schema = None
  1186. permission_classes = (permissions.IsAdminUser,)
  1187. def delete(self, request, uid, format=None):
  1188. revoke(uid, terminate=True)
  1189. return Response(status=status.HTTP_204_NO_CONTENT)