diff --git a/mcserver/admin.py b/mcserver/admin.py index 61cd1f7..e80671e 100644 --- a/mcserver/admin.py +++ b/mcserver/admin.py @@ -1,6 +1,8 @@ from django.contrib import admin +from django.contrib.admin.models import LogEntry +from django.contrib.auth.admin import UserAdmin from django.shortcuts import render, redirect -from django.contrib.auth.admin import UserAdmin, GroupAdmin + from mcserver.models import ( User, Session, @@ -17,14 +19,14 @@ SubjectTags, TrialTags ) + from django.contrib.auth.models import Group from django.contrib.auth.admin import UserAdmin, GroupAdmin from django.contrib.admin.models import LogEntry from datetime import timedelta - -#admin.site.unregister(Group) -#admin.site.register(Group, GroupAdmin) +# admin.site.unregister(Group) +# admin.site.register(Group, GroupAdmin) @admin.register(User) @@ -165,6 +167,7 @@ class SubjectAdmin(admin.ModelAdmin): ) raw_id_fields = ('user',) + @admin.register(SubjectTags) class SubjectTagsAdmin(admin.ModelAdmin): search_fields = ['tag', 'subject__name'] @@ -191,7 +194,7 @@ class ResetPasswordAdmin(admin.ModelAdmin): @admin.register(LogEntry) class LogEntryAdmin(admin.ModelAdmin): - # to have a date-based drilldown navigation in the admin page + # to have a date-based drill-down navigation in the admin page date_hierarchy = 'action_time' # to filter the resultes by users, content types and action flags diff --git a/mcserver/authentication.py b/mcserver/authentication.py index b1575a0..0f4e01f 100644 --- a/mcserver/authentication.py +++ b/mcserver/authentication.py @@ -40,4 +40,4 @@ def authenticate_credentials(self, key): Token.objects.create(user=token.user) raise AuthenticationFailed("Token has expired") - return (token.user, token) + return token.user, token diff --git a/mcserver/constants.py b/mcserver/constants.py index a9a0f87..21f44ff 100644 --- a/mcserver/constants.py +++ b/mcserver/constants.py @@ -5,7 +5,7 @@ class ResultTag(Enum): CALIBRATION_IMAGE = "calibration-img" CAMERA_CALIBRATION_OPTS = "calibration_parameters_options" - IK_RESULTS= "ik_results" + IK_RESULTS = "ik_results" MARKER_DATA = "marker_data" OPENSIM_MODEL = "opensim_model" POSE_PICKLE = "pose_pickle" diff --git a/mcserver/customEmailDevice.py b/mcserver/customEmailDevice.py index e6d0212..0a54a61 100644 --- a/mcserver/customEmailDevice.py +++ b/mcserver/customEmailDevice.py @@ -5,6 +5,7 @@ from django.template import Template, Context from django.template.loader import get_template + class CustomEmailDevice(EmailDevice): def generate_challenge(self, extra_context=None): @@ -25,8 +26,8 @@ def generate_challenge(self, extra_context=None): send_mail(settings.OTP_EMAIL_SUBJECT, strip_tags(body), settings.OTP_EMAIL_SENDER, - [self.email or self.user.email] - ,html_message=body) + [self.email or self.user.email], + html_message=body) message = "sent by email" diff --git a/mcserver/models.py b/mcserver/models.py index e94491f..cc1f147 100644 --- a/mcserver/models.py +++ b/mcserver/models.py @@ -1,23 +1,15 @@ -import json - -from django.db import models -from django.contrib.auth.models import AbstractUser -from django.core.validators import MinValueValidator, MaxValueValidator import os import uuid -import base64 -import pathlib from http import HTTPStatus -from django.utils import timezone + +from django.contrib.auth.models import AbstractUser from django.core.exceptions import ValidationError +from django.db import models from django.db.models.signals import post_save -from django.contrib.auth.signals import user_logged_in from django.dispatch import receiver -from rest_framework.authtoken.models import Token +from django.utils import timezone from django.utils.translation import gettext as _ -from rest_framework import status - -from django.conf import settings +from django_otp.plugins.otp_email.models import EmailDevice def random_filename(instance, filename): @@ -185,10 +177,11 @@ def is_public(self): def get_user(self): return self.trial.get_user() + class Result(models.Model): trial = models.ForeignKey(Trial, blank=False, null=False, on_delete=models.CASCADE) device_id = models.CharField(max_length=36, blank=True, null=True) - media = models.FileField(blank=True, null=True, upload_to=random_filename,max_length=500) + media = models.FileField(blank=True, null=True, upload_to=random_filename, max_length=500) tag = models.CharField(max_length=32, blank=True, null=True) meta = models.JSONField(blank=True, null=True) created_at = models.DateTimeField(auto_now_add=True, db_index=True) @@ -217,9 +210,12 @@ def commit(cls, trial, device_id, tag, media_path, meta=None): ) @classmethod - def reset(cls, trial, tag=None, selected=[]): + def reset(cls, trial, tag=None, selected=None): """ Deletes selected results, or all for trial with the tag """ + if selected is None: + selected = [] + if selected: cls.objects.filter(id__in=selected).delete() elif tag: @@ -248,15 +244,12 @@ class ResetPassword(models.Model): email = models.CharField(max_length=255) datetime = models.DateField(default=timezone.now) -from django_otp.plugins.otp_email.models import EmailDevice -from django.template.loader import render_to_string -from mcserver.customEmailDevice import CustomEmailDevice @receiver(post_save, sender=User) def create_profile(sender, instance, created, **kwargs): """Create a matching profile whenever a user object is created.""" if created: - device = EmailDevice(user = instance, name = "default e-mail") + device = EmailDevice(user=instance, name="default e-mail") device.save() @@ -423,7 +416,7 @@ class AnalysisResult(models.Model): help_text='Trial function was called with. Set automatically.', ) response = models.JSONField( - 'Response', default=dict, help_text='Data function responsed with.' + 'Response', default=dict, help_text='Data function responded with.' ) result = models.ForeignKey( to=Result, @@ -436,7 +429,7 @@ class AnalysisResult(models.Model): 'Status', choices=[(status.value, status.phrase) for status in list(HTTPStatus)], default=HTTPStatus.OK.value, - help_text='Status code function responsed with.' + help_text='Status code function responded with.' ) state = models.CharField( 'Invokation state', diff --git a/mcserver/serializers.py b/mcserver/serializers.py index e2ef10d..136bb2b 100644 --- a/mcserver/serializers.py +++ b/mcserver/serializers.py @@ -1,6 +1,10 @@ import json + +from django.db.models import Prefetch, Q +from django.utils.translation import gettext as _ from rest_framework import serializers -from rest_framework import pagination +from rest_framework.validators import UniqueValidator + from mcserver.models import ( Session, User, @@ -15,9 +19,6 @@ SubjectTags, TrialTags ) -from rest_framework.validators import UniqueValidator -from django.db.models import Prefetch, Q -from django.utils.translation import gettext as _ class UserSerializer(serializers.ModelSerializer): @@ -88,8 +89,8 @@ def create(self, validated_data): class Meta: model = User - fields = ('id', 'username', 'first_name', 'last_name', 'email', 'country', 'institution', 'profession', 'reason', - 'website', 'newsletter') + fields = ('id', 'username', 'first_name', 'last_name', 'email', 'country', 'institution', 'profession', + 'reason', 'website', 'newsletter') class ProfilePictureSerializer(serializers.ModelSerializer): @@ -122,16 +123,18 @@ class NewPasswordSerializer(serializers.ModelSerializer): class Meta: model = User - fields = ('password','token',) + fields = ('password', 'token',) # Serializers define the API representation. class VideoSerializer(serializers.ModelSerializer): video_url = serializers.CharField(max_length=256, required=False) + class Meta: model = Video - fields = ['id', 'trial', 'device_id', 'video', 'video_url', 'video_thumb', 'parameters', 'created_at', 'updated_at'] + fields = ['id', 'trial', 'device_id', 'video', 'video_url', 'video_thumb', 'parameters', 'created_at', + 'updated_at'] # Serializers define the API representation. diff --git a/mcserver/settings.py b/mcserver/settings.py index 0ced436..e590293 100644 --- a/mcserver/settings.py +++ b/mcserver/settings.py @@ -12,12 +12,14 @@ from pathlib import Path from decouple import config +from datetime import timedelta +from celery.schedules import crontab + import os.path # Build paths inside the project like this: BASE_DIR / 'subdir'. BASE_DIR = Path(__file__).resolve().parent.parent - # Quick-start development settings - unsuitable for production # See https://docs.djangoproject.com/en/3.1/howto/deployment/checklist/ @@ -31,7 +33,7 @@ PROTOCOL = config("PROTOCOL", "http") HOST_URL = "{}://{}".format(PROTOCOL, HOST) -ALLOWED_HOSTS = [HOST,"*"] +ALLOWED_HOSTS = [HOST, "*"] CORS_ALLOW_ALL_ORIGINS = True @@ -85,7 +87,7 @@ 'django.middleware.common.CommonMiddleware', 'django.middleware.csrf.CsrfViewMiddleware', 'django.contrib.auth.middleware.AuthenticationMiddleware', - 'django_otp.middleware.OTPMiddleware', + 'django_otp.middleware.OTPMiddleware', 'django.contrib.messages.middleware.MessageMiddleware', 'django.middleware.clickjacking.XFrameOptionsMiddleware', ] @@ -110,21 +112,20 @@ WSGI_APPLICATION = 'mcserver.wsgi.application' - # Database # https://docs.djangoproject.com/en/3.1/ref/settings/#databases -DATABASES = { - 'default': { - 'ENGINE': 'django.db.backends.sqlite3', - 'NAME': BASE_DIR / 'data/db.sqlite3', - } -} +# DATABASES = { +# 'default': { +# 'ENGINE': 'django.db.backends.sqlite3', +# 'NAME': BASE_DIR / 'data/db.sqlite3', +# } +# } DATABASES = { 'default': { 'ENGINE': 'django.db.backends.postgresql', - 'NAME': config("DB_NAME",default="opencap"), + 'NAME': config("DB_NAME", default="opencap"), 'USER': config("DB_USER"), 'PASSWORD': config("DB_PASS"), 'HOST': config("DB_HOST"), @@ -132,7 +133,6 @@ } } - # Password validation # https://docs.djangoproject.com/en/3.1/ref/settings/#auth-password-validators @@ -151,7 +151,6 @@ }, ] - # Internationalization # https://docs.djangoproject.com/en/3.1/topics/i18n/ @@ -165,7 +164,6 @@ USE_TZ = True - # Static files (CSS, JavaScript, Images) # https://docs.djangoproject.com/en/3.1/howto/static-files/ @@ -183,15 +181,15 @@ AWS_S3_REGION_NAME = config("REGION", default="us-west-2") AWS_S3_ENDPOINT_URL = config('AWS_S3_ENDPOINT_URL', default=None) -#AWS_S3_OBJECT_PARAMETERS = {'CacheControl': 'max-age=86400'} +# AWS_S3_OBJECT_PARAMETERS = {'CacheControl': 'max-age=86400'} # s3 static settings # STATIC_URL = f'https://{AWS_S3_CUSTOM_DOMAIN}/' STATICFILES_STORAGE = 'storages.backends.s3boto3.S3Boto3Storage' -#MEDIA_LOCATION = 'media' -#MEDIA_URL = f'https://{AWS_S3_CUSTOM_DOMAIN}/{MEDIA_LOCATION}/' -#DEFAULT_FILE_STORAGE = 'storages.backends.s3boto3.S3Boto3Storage' +# MEDIA_LOCATION = 'media' +# MEDIA_URL = f'https://{AWS_S3_CUSTOM_DOMAIN}/{MEDIA_LOCATION}/' +# DEFAULT_FILE_STORAGE = 'storages.backends.s3boto3.S3Boto3Storage' MEDIA_ROOT = os.path.join(BASE_DIR, 'media') ARCHIVES_ROOT = config('ARCHIVES_ROOT', default=os.path.join(MEDIA_ROOT, 'archives')) @@ -209,8 +207,6 @@ LOGO_LINK = "https://app.opencap.ai/images/opencap-logo.png" -from datetime import timedelta - AUTH_TOKEN_VALIDITY = timedelta(days=90) OTP_EMAIL_SENDER = DEFAULT_FROM_EMAIL @@ -220,7 +216,6 @@ OTP_EMAIL_SUBJECT = "" OTP_EMAIL_BODY_TEMPLATE = "" - # Sentry support SENTRY_DSN = config('SENTRY_DSN', default='') @@ -234,7 +229,7 @@ def strip_sensitive_data(event, hint): """ This function removes the DisallowedHost errors from - the Sentry logs for avoiding excedding the quota. + the Sentry logs for avoiding exceeding the quota. """ if 'log_record' in hint: if hint['log_record'].name == 'django.security.DisallowedHost': @@ -266,8 +261,6 @@ def strip_sensitive_data(event, hint): CELERY_BEAT_SCHEDULER = 'redbeat.RedBeatScheduler' CELERY_IMPORTS = ['mcserver.tasks'] -from celery.schedules import crontab - CELERY_BEAT_SCHEDULE = { 'cleanup_trashed_sessions': { 'task': 'mcserver.tasks.cleanup_trashed_sessions', @@ -301,7 +294,9 @@ def strip_sensitive_data(event, hint): # }, } -GRAPH_MODELS ={ +GRAPH_MODELS = { 'all_applications': True, 'graph_models': True, -} \ No newline at end of file +} + +DEFAULT_AUTO_FIELD = 'django.db.models.AutoField' diff --git a/mcserver/tasks.py b/mcserver/tasks.py index 009d0cb..518d030 100644 --- a/mcserver/tasks.py +++ b/mcserver/tasks.py @@ -1,13 +1,13 @@ -import os import json -import requests +import os +from datetime import timedelta from http import HTTPStatus + +import requests +from celery import shared_task from django.conf import settings -from django.core.files import File from django.core.files.base import ContentFile -from celery import shared_task from django.utils import timezone -from datetime import timedelta from mcserver.models import ( DownloadLog, @@ -74,6 +74,7 @@ def download_session_archive(self, session_id, user_id=None): else: print(e) + @shared_task(bind=True) def download_subject_archive(self, subject_id, user_id): """ This task is responsible for asynchronous subject archive download @@ -175,7 +176,7 @@ def invoke_aws_lambda_function(self, user_id, function_id, data): analysis_result.response = function_response analysis_result.save(update_fields=['result', 'status', 'state', 'response']) - # Crreate analysis dashboard if available + # Create analysis dashboard if available try: AnalysisDashboard.objects.get(user_id=user_id, function_id=function_id) except AnalysisDashboard.DoesNotExist: diff --git a/mcserver/urls.py b/mcserver/urls.py index beb3478..d22b952 100644 --- a/mcserver/urls.py +++ b/mcserver/urls.py @@ -14,7 +14,13 @@ 2. Add a URL to urlpatterns: path('blog/', include('blog.urls')) """ from django.contrib import admin +from django.http import HttpResponse from django.urls import include, path +from drf_yasg import openapi +from drf_yasg.views import get_schema_view +from rest_framework import permissions +from rest_framework import routers + from mcserver.views import ( SessionViewSet, VideoViewSet, @@ -46,18 +52,6 @@ UserUpdate, UpdateProfilePicture ) -from rest_framework import routers, serializers, viewsets -from rest_framework.authtoken.views import obtain_auth_token - -from django_otp.forms import OTPAuthenticationForm -from django.http import HttpResponse - -from django.conf.urls.static import static -from rest_framework_swagger.views import get_swagger_view -from drf_yasg.views import get_schema_view -from drf_yasg import openapi -from rest_framework import permissions -from django.conf import settings schema_view = get_schema_view( openapi.Info( @@ -80,10 +74,10 @@ router.register(r'analysis-dashboards', AnalysisDashboardViewSet, "analysis-dashboard") urlpatterns = [ -# path('session/', new_session), + # path('session/', new_session), path('', include(router.urls)), path("health/", lambda x: HttpResponse("OK"), name="health"), -# path('session//status/', status), + # path('session//status/', status), path('login/', CustomAuthToken.as_view()), path('verify/', verify), path('set-institutional-use/', set_institutional_use), @@ -137,6 +131,7 @@ AnalysisFunctionsStatesForTrialsAPIView.as_view(), name='analysis-results-statuses-for-trials' ), + path('subject-tags//get_tags_subject/', SubjectTagViewSet.as_view({'get': 'get_tags_subject'}), name='get_tags_subject'), path('trial-tags//get_tags_trial/', TrialTagViewSet.as_view({'get': 'get_tags_trial'}), diff --git a/mcserver/views.py b/mcserver/views.py index 417a870..16444f7 100644 --- a/mcserver/views.py +++ b/mcserver/views.py @@ -1,31 +1,48 @@ -import boto3 -import uuid -import sys -import os -import qrcode import json -import time +import os import platform -import traceback import socket - +import sys +import time +import traceback +import uuid from datetime import datetime, timedelta -from django.shortcuts import get_object_or_404 +import boto3 +import qrcode +from django.conf import settings from django.contrib.auth import login +from django.contrib.auth.models import AnonymousUser from django.core.files.base import ContentFile -from django.utils.timezone import now -from django.utils import timezone -from django.http import Http404 +from django.core.mail import EmailMessage +from django.db.models import Count from django.db.models import Q -from django.utils.translation import gettext as _ from django.http import FileResponse -from django.db.models import Count -from django.db.models import F -from django.views.decorators.csrf import csrf_exempt -from django.conf import settings -from django.core.mail import EmailMessage +from django.http import Http404 +from django.shortcuts import get_object_or_404 from django.template.loader import render_to_string +from django.utils import timezone +from django.utils.timezone import now +from django.utils.translation import gettext as _ +from django.views.decorators.csrf import csrf_exempt +from drf_yasg import openapi +from rest_framework import permissions +from rest_framework import status +from rest_framework import viewsets +from rest_framework.authtoken.models import Token +from rest_framework.authtoken.views import ObtainAuthToken +from rest_framework.decorators import action +from rest_framework.decorators import api_view, renderer_classes +from rest_framework.exceptions import ValidationError, NotAuthenticated, NotFound, PermissionDenied, APIException +from rest_framework.generics import ListAPIView +from rest_framework.permissions import AllowAny +from rest_framework.permissions import IsAuthenticated +from rest_framework.renderers import JSONRenderer, TemplateHTMLRenderer +from rest_framework.response import Response +from rest_framework.reverse import reverse +from rest_framework.views import APIView + +from drf_yasg.utils import swagger_auto_schema from mcserver.models import ( Session, @@ -41,7 +58,6 @@ AnalysisFunction, AnalysisResult, AnalysisResultState, - AnalysisDashboardTemplate, AnalysisDashboard, ) from mcserver.serializers import ( @@ -58,41 +74,31 @@ NewPasswordSerializer, AnalysisFunctionSerializer, AnalysisResultSerializer, - AnalysisDashboardTemplateSerializer, AnalysisDashboardSerializer, ProfilePictureSerializer, UserInstitutionalUseSerializer, + TagSerializer, + ValidSessionLightSerializer, SubjectTagSerializer, TrialTagSerializer ) -from mcserver.utils import send_otp_challenge -from mcserver.zipsession import downloadAndZipSession, downloadAndZipSubject from mcserver.tasks import ( download_session_archive, download_subject_archive, invoke_aws_lambda_function ) - -from rest_framework.exceptions import ValidationError, NotAuthenticated, NotFound, PermissionDenied, APIException -from rest_framework.permissions import IsAuthenticated -from rest_framework.decorators import action -from rest_framework import permissions -from rest_framework.response import Response -from rest_framework.reverse import reverse -from rest_framework.views import APIView -from rest_framework.authtoken.views import ObtainAuthToken -from rest_framework.authtoken.models import Token -from rest_framework.permissions import AllowAny -from rest_framework.generics import ListAPIView -from rest_framework import viewsets -from rest_framework.decorators import api_view, renderer_classes -from rest_framework.renderers import JSONRenderer, TemplateHTMLRenderer -from rest_framework import status - +from mcserver.utils import send_otp_challenge +from mcserver.zipsession import downloadAndZipSession, downloadAndZipSubject sys.path.insert(0, '/code/mobilecap') + class IsOwner(permissions.BasePermission): + """ + Allow owners of an object to perform operations. + + A user is 'owner' if it is authenticated and the user associated to the object. + """ def has_permission(self, request, view): if not request.user.is_authenticated: return False @@ -101,28 +107,52 @@ def has_permission(self, request, view): def has_object_permission(self, request, view, obj): return obj.get_user() == request.user + class IsAdmin(permissions.BasePermission): + """ + Allow admins to perform operations. + + A user is admin if it belongs to the 'admin' group. + """ def has_permission(self, request, view): return request.user.groups.filter(name='admin').exists() def has_object_permission(self, request, view, obj): return self.has_permission(request, view) + class IsBackend(permissions.BasePermission): + """ + Allow backend to perform operations. + + A user is backend if it belongs to the 'backend' group. + """ def has_permission(self, request, view): return request.user.groups.filter(name='backend').exists() def has_object_permission(self, request, view, obj): return self.has_permission(request, view) + class IsPublic(permissions.BasePermission): + """ + Allows public users to perform operations. + + A method is public if it is a GET operation and the object retrieved is marked as 'public'. + """ def has_permission(self, request, view): return request.method == "GET" def has_object_permission(self, request, view, obj): return obj.is_public() + class AllowPublicCreate(permissions.BasePermission): + """ + Allows public users to create new resources or update existing ones. + + Permission is granted for POST (create) and PATCH (update) methods. + """ def has_permission(self, request, view): # create new or update existing video return (request.method == "POST") or (request.method == "PATCH") @@ -130,7 +160,14 @@ def has_permission(self, request, view): def has_object_permission(self, request, view, obj): return self.has_permission(request, view) + def setup_eager_loading(get_queryset): + """ + Decorator function to enable eager loading for a queryset. + + This decorator modifies the `get_queryset` method of a view to + include related objects in a single query, optimizing database access. + """ def decorator(self): queryset = get_queryset(self) queryset = self.get_serializer_class().setup_eager_loading(queryset) @@ -138,9 +175,14 @@ def decorator(self): return decorator -#from utils import switchCalibrationForCamera def get_client_ip(request): + """ + Retrieves the client's IP address from the given request. + + This function checks the `HTTP_X_FORWARDED_FOR` header first (which may be + set by proxies), and if not present, falls back to the `REMOTE_ADDR` header. + """ x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR') if x_forwarded_for: ip = x_forwarded_for.split(',')[0] @@ -148,7 +190,14 @@ def get_client_ip(request): ip = request.META.get('REMOTE_ADDR') return ip + def get_client_hostname(request): + """ + Retrieves the hostname corresponding to the client's IP address. + + Uses the IP address from `get_client_ip()` and performs a reverse DNS lookup + to get the hostname. If the lookup fails, it returns `None`. + """ ip = get_client_ip(request) try: hostname = socket.gethostbyaddr(ip) @@ -156,39 +205,79 @@ def get_client_hostname(request): except socket.herror: return None + def zipdir(path, ziph): + """ + Compresses a directory into a zip archive. + + Traverses the directory tree starting at the specified path and adds all files + to the provided zip file handle. Preserves the directory structure in the archive. + """ # ziph is zipfile handle for root, dirs, files in os.walk(path): for file in files: - ziph.write(os.path.join(root, file), - os.path.relpath(os.path.join(root, file), + ziph.write(os.path.join(root, file), + os.path.relpath(os.path.join(root, file), os.path.join(path, '..'))) class SessionViewSet(viewsets.ModelViewSet): + """ + A view set for viewing and editing session objects. + + """ serializer_class = SessionSerializer - permission_classes = [IsPublic | ((IsOwner | IsAdmin | IsBackend))] + permission_classes = [IsPublic | (IsOwner | IsAdmin | IsBackend)] @setup_eager_loading def get_queryset(self): """ - This view should return a list of all the sessions - for the currently authenticated user. + Retrieves the queryset of sessions available to the current user. """ user = self.request.user if user.is_authenticated and user.id == 1: return Session.objects.all().order_by("-created_at") return Session.objects.filter(Q(user__id=user.id) | Q(public=True)).order_by("-created_at") + @swagger_auto_schema( + operation_summary="API Health Check", + responses={ + 200: openapi.Response("Success - API health retrieved successfully."), + } + ) @action(detail=False) def api_health_check(self, request): + """ + Check the health of the API. + """ return Response({"status": "True"}) - @action( - detail=True, - methods=["get", "post"], + @swagger_auto_schema( + method="post", + operation_summary="Update Calibration Data", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + 'calibration': openapi.Schema( + type=openapi.TYPE_OBJECT, + description="A dictionary containing calibration data for the session.", + additionalProperties=openapi.Schema(type=openapi.TYPE_STRING) + ), + }, + required=['calibration'], + description="Calibration data to be updated." + ), + responses={ + 200: openapi.Response("Success - Calibration data updated successfully."), + 400: openapi.Response("Bad Request - Invalid session data."), + 404: openapi.Response("Not Found - Session not found."), + }, ) + @action(detail=True, methods=["get", "post"], ) def calibration(self, request, pk): + """ + Update calibration data for a specific session. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -220,8 +309,19 @@ def calibration(self, request, pk): "data": request.data, }) + @swagger_auto_schema( + operation_summary="Get Number of Calibrated Cameras", + responses={ + 200: openapi.Response("Success - Number of calibrated cameras retrieved successfully."), + 400: openapi.Response("Bad Request - Invalid Session data."), + 404: openapi.Response("Not Found - Session not found."), + } + ) @action(detail=True) def get_n_calibrated_cameras(self, request, pk): + """ + Retrieve the number of calibrated cameras for a specific session. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -256,7 +356,7 @@ def get_n_calibrated_cameras(self, request, pk): last_calibration_trial_num_videos = Video.objects.filter(trial=last_calibration_trial).count() else: error_message = 'Sorry, there is no calibration trial for this session.' \ - 'Maybe it was created from a session that was remove.' + 'Maybe it was created from a session that was remove.' except Http404: if settings.DEBUG: @@ -276,8 +376,26 @@ def get_n_calibrated_cameras(self, request, pk): 'data': last_calibration_trial_num_videos }) + @swagger_auto_schema( + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + required=['sessionNewName'], + properties={ + 'sessionNewName': openapi.Schema(type=openapi.TYPE_STRING, description="New name for the session"), + }, + description="Object containing the new session name." + ), + responses={ + 200: openapi.Response("Success - Session renamed successfully."), + 400: openapi.Response("Bad Request - Invalid session data."), + 404: openapi.Response("Not Found - Session not found."), + }, + ) @action(detail=True, methods=['post']) def rename(self, request, pk): + """ + Rename a specific session. + """ # Get session. session = get_object_or_404(Session.objects.all(), pk=pk) @@ -319,7 +437,19 @@ def rename(self, request, pk): 'data': serializer.data }) + @swagger_auto_schema( + operation_summary="Retrieve a session", + responses={ + 200: openapi.Response("Success - Session retrieved successfully."), + 401: openapi.Response("Unauthorized - User must be authenticated."), + 403: openapi.Response("Forbidden - Authentication is required."), + 404: openapi.Response("Not Found - Session not found."), + }, + ) def retrieve(self, request, pk=None): + """ + Retrieve a specific session. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -352,14 +482,81 @@ def retrieve(self, request, pk=None): return Response(serializer.data) - - - @action( - detail=False, - methods=["get", "post"], + @swagger_auto_schema( + method="get", + operation_summary="Retrieve valid sessions.", + manual_parameters=[ + openapi.Parameter('quantity', openapi.IN_QUERY, + description="Number of sessions to retrieve.", + type=openapi.TYPE_INTEGER), + openapi.Parameter('start', openapi.IN_QUERY, + description="Starting index for session retrieval.", + type=openapi.TYPE_INTEGER), + openapi.Parameter('subject_id', openapi.IN_QUERY, description="Filter sessions by subject ID.", type=openapi.TYPE_INTEGER), + openapi.Parameter('sort', openapi.IN_QUERY, description="Sort options for the sessions.", type=openapi.TYPE_ARRAY, items=openapi.Items(type=openapi.TYPE_STRING)), + openapi.Parameter('sort_desc', openapi.IN_QUERY, description="Sort descending flags for each sort field.", type=openapi.TYPE_ARRAY, items=openapi.Items(type=openapi.TYPE_BOOLEAN)), + openapi.Parameter('include_trashed', openapi.IN_QUERY, description="Include trashed sessions in the results.", type=openapi.TYPE_BOOLEAN), + openapi.Parameter('only_trashed', openapi.IN_QUERY, description="Retrieve only trashed sessions.", type=openapi.TYPE_BOOLEAN), + ], + responses={ + 200: openapi.Response("Success - Session retrieved and validated successfully.", + schema=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + 'sessions': openapi.Schema( + type=openapi.TYPE_ARRAY, + items=openapi.Schema(type=openapi.TYPE_OBJECT) + ), + 'total': openapi.Schema(type=openapi.TYPE_INTEGER, description="Total count of valid sessions.") + }, + ) + ), + 400: openapi.Response("Bad Request - Invalid subject data."), + 404: openapi.Response("Not Found - Session not found."), + } ) + @swagger_auto_schema( + method="post", + operation_summary="Validate and retrieve user sessions.", + responses={ + 200: openapi.Response("Success - Session retrieved and validated successfully.", + schema=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + 'sessions': openapi.Schema( + type=openapi.TYPE_ARRAY, + items=openapi.Schema(type=openapi.TYPE_OBJECT) + ), + 'total': openapi.Schema(type=openapi.TYPE_INTEGER, description="Total count of valid sessions.") + }, + )), + 400: openapi.Response("Bad Request - Invalid subject data."), + 404: openapi.Response("Not Found - Session not found."), + }, + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + required=["quantity"], + properties={ + 'quantity': openapi.Schema(type=openapi.TYPE_INTEGER, + description="Number of sessions to retrieve. Set to -1 to retrieve all."), + 'start': openapi.Schema(type=openapi.TYPE_INTEGER, description="Starting index for session retrieval."), + 'subject_id': openapi.Schema(type=openapi.TYPE_INTEGER, description="Filter sessions by subject ID."), + 'sort': openapi.Schema(type=openapi.TYPE_ARRAY, items=openapi.Schema(type=openapi.TYPE_STRING), + description="Sort options for the sessions."), + 'sort_desc': openapi.Schema(type=openapi.TYPE_ARRAY, items=openapi.Schema(type=openapi.TYPE_BOOLEAN), + description="Sort descending flags for each sort field."), + 'include_trashed': openapi.Schema(type=openapi.TYPE_BOOLEAN, + description="Include trashed sessions in the results."), + 'only_trashed': openapi.Schema(type=openapi.TYPE_BOOLEAN, + description="Retrieve only trashed sessions."), + } + ), + ) + @action(detail=False, methods=["get", "post"], ) def valid(self, request): - from .serializers import ValidSessionLightSerializer + """ + Validate and retrieve user sessions based on various filters and sorting options. + """ try: # print(request.data) include_trashed = request.data.get('include_trashed', False) is True @@ -372,9 +569,9 @@ def valid(self, request): else: quantity = request.data['quantity'] start = 0 if 'start' not in request.data else request.data['start'] - # Note the use of `get_queryset()` instead of `self.queryset` + # Note the use of `get_queryset()` instead of `self.queryset` (Confusing) sessions = self.get_queryset() \ - .annotate(trial_count=Count('trial'))\ + .annotate(trial_count=Count('trial')) \ .filter(trial_count__gte=1, user=request.user) if only_trashed: @@ -400,7 +597,7 @@ def valid(self, request): trials_count=Count( 'trial', filter=~Q(trial__name='calibration') & ~(Q(trial__name='neutral') & ~Q(trial__status='done')), - )) + )) sort_options = { 'name': 'subject__name', 'trials_count': 'trials_count', @@ -409,7 +606,8 @@ def valid(self, request): } sessions = sessions.order_by( - *[('-' if sort_desc[i] else '')+sort_options[x] for i, x in enumerate(sort_by) if x in sort_options], '-id') + *[('-' if sort_desc[i] else '') + sort_options[x] for i, x in enumerate(sort_by) if + x in sort_options], '-id') sessions_count = sessions.count() # If quantity is not -1, retrieve only last n sessions. @@ -433,8 +631,21 @@ def valid(self, request): return Response({'sessions': serializer.data, 'total': sessions_count}) return Response(serializer.data) + @swagger_auto_schema( + operation_summary="Permanently remove a session", + responses={ + 204: openapi.Response("Deleted - Session deleted successfully."), + 401: openapi.Response("Unauthorized - User must be authenticated."), + 403: openapi.Response("Forbidden - Authentication is required."), + 404: openapi.Response("Not Found - Session not found."), + 500: openapi.Response("Internal Server Error - Could not remove the session."), + }, + ) @action(detail=True, methods=['post']) def permanent_remove(self, request, pk): + """ + Permanently remove a specific session by its ID (UUID). + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -465,8 +676,30 @@ def permanent_remove(self, request, pk): return Response({}) + @swagger_auto_schema( + operation_summary="Move a session to the trash", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + "pk": openapi.Schema( + type=openapi.TYPE_STRING, + description="UUID of the session to be moved to the trash.", + ), + }, + required=["pk"], + description="JSON payload with the UUID of the session.", + ), + responses={ + 200: openapi.Response("Success - Session trashed successfully."), + 404: openapi.Response("Not Found - Session not found."), + 500: openapi.Response("Internal Server Error - Could not trash the session."), + }, + ) @action(detail=True, methods=['post']) def trash(self, request, pk): + """ + Move a specific session to the trash by marking it as 'trashed'. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -492,8 +725,19 @@ def trash(self, request, pk): return Response(serializer.data) + @swagger_auto_schema( + operation_summary="Restore a trashed session", + responses={ + 200: openapi.Response("Success - Session restored successfully."), + 404: openapi.Response("Not Found - Session not found."), + 500: openapi.Response("Internal Server Error - Could not restore the session."), + }, + ) @action(detail=True, methods=['post']) def restore(self, request, pk): + """ + Restore a specific session from the trash. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -519,11 +763,19 @@ def restore(self, request, pk): return Response(serializer.data) - - ## New session GET '/new/' - # Creates a new session, returns session id and the QR code + @swagger_auto_schema( + operation_summary="Create a new session and generate a QR code for it.", + responses={ + 200: openapi.Response("Success - Session and QR created successfully."), + 400: openapi.Response("Bad Request - Invalid request data."), + 500: openapi.Response("Internal Server Error - Could not restore the session."), + } + ) @action(detail=False) def new(self, request): + """ + Create a new session and generate a QR code for it. + """ try: session = Session() @@ -560,21 +812,32 @@ def new(self, request): raise APIException(_("session_create_error")) return Response(serializer.data) - - ## Get and send QR code + + @swagger_auto_schema( + operation_summary="Retrieve QR code for a session.", + responses={ + 200: openapi.Response("Success - Session QR retrieved successfully."), + 400: openapi.Response("Bad Request - Invalid session data."), + 404: openapi.Response("Not Found - Session not found."), + 500: openapi.Response("Internal Server Error - Could not retrieve the QR."), + } + ) @action(detail=True) def get_qr(self, request, pk): + """ + Retrieve the QR code for the session. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) session = get_object_or_404(Session, pk=pk, user=request.user) - + # get the QR code from the database if session.qrcode: qr = session.qrcode elif session.meta and 'sessionWithCalibration' in session.meta: - sessionWithCalibration = Session.objects.get(pk = str(session.meta['sessionWithCalibration']['id'])) + sessionWithCalibration = Session.objects.get(pk=str(session.meta['sessionWithCalibration']['id'])) qr = sessionWithCalibration.qrcode s3_client = boto3.client( @@ -601,12 +864,21 @@ def get_qr(self, request, pk): raise APIException(_("qr_retrieve_error")) return Response(res) - - ## New session GET '/new_subject/' - # Creates a new sessionm leaving metadata on previous session. Used to avoid - # re-connecting and re-calibrating cameras with every new subject. + + @swagger_auto_schema( + operation_summary="Create a new session for a new subject.", + responses={ + 200: openapi.Response("Success - Session created successfully."), + 400: openapi.Response("Bad Request - Invalid session data."), + 404: openapi.Response("Not Found - Session not found."), + 500: openapi.Response("Internal Server Error - Could not create the session."), + } + ) @action(detail=True) def new_subject(self, request, pk): + """ + Create a new session for a new subject. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -634,8 +906,8 @@ def new_subject(self, request, pk): sessionNew.meta = {} sessionNew.meta["sessionWithCalibration"] = { - "id": sessionWithCalibration - } + "id": sessionWithCalibration + } sessionNew.save() # tell the old session to go to the new session - phones will connect to this new session @@ -668,12 +940,19 @@ def new_subject(self, request, pk): return Response(serializer.data) - def get_permissions(self): if self.action == 'status' or self.action == 'get_status': return [AllowAny(), ] return super(SessionViewSet, self).get_permissions() - + + @swagger_auto_schema( + operation_summary="Get the status of the session.", + responses={ + 200: openapi.Response("Success - Session status retrieved successfully."), + 404: openapi.Response("Not Found - Session not found."), + 500: openapi.Response("Internal Server Error - Could not retrieve the session status."), + }, + ) def get_status(self, request, pk): if pk == 'undefined': raise NotFound(_("session_uuid_not_valid") % {"uuid": str(pk)}) @@ -685,7 +964,7 @@ def get_status(self, request, pk): trials = session.trial_set.order_by("-created_at") trial = None - status = "ready" # if no trials then "ready" (equivalent to trial_status = done) + status = "ready" # if no trials then "ready" (equivalent to trial_status = done) # If there is at least one trial, check it's status if trials.count(): @@ -694,10 +973,10 @@ def get_status(self, request, pk): # if trial_status == 'done' then session ready again if trial and trial.status == "done": status = "ready" - + # if trial_status == 'recording' then just continue and return 'recording' # otherwise recording is done and check processing - if trial and (trial.status in ["stopped","processing"]): + if trial and (trial.status in ["stopped", "processing"]): # if not all videos uploaded then the status is 'uploading' # if results are not ready then processing # otherwise it's ready again @@ -730,21 +1009,20 @@ def get_status(self, request, pk): if videos.count() > 0: video_url = reverse('video-detail', kwargs={'pk': videos[0].id}) trial_url = reverse('trial-detail', kwargs={'pk': trial.id}) if trial else None - + # tell phones to pair with a new session if session.meta and "startNewSession" in session.meta: newSessionURL = "{}/sessions/{}/status/".format(settings.HOST_URL, session.meta['startNewSession']['id']) else: newSessionURL = None - + if session.meta and "settings" in session.meta and "framerate" in session.meta['settings']: frameRate = int(session.meta['settings']['framerate']) else: frameRate = 60 - if trial and (trial.name in {'calibration','neutral'}): + if trial and (trial.name in {'calibration', 'neutral'}): frameRate = 30 - res = { "status": status, "trial": trial_url, @@ -757,56 +1035,83 @@ def get_status(self, request, pk): if "ret_session" in request.GET: res["session"] = SessionSerializer(session, many=False).data - + return res - @action(detail=True) def get_presigned_url(self, request, pk): + """ + Generates a presigned URL for uploading a file to S3. + """ s3_client = boto3.client( 's3', aws_access_key_id=settings.AWS_ACCESS_KEY_ID, aws_secret_access_key=settings.AWS_SECRET_ACCESS_KEY, ) - - if request.data and request.data.get('fileName'): - fileName = '-' + request.data.get('fileName') # for result uploading - matching old way - else: # default: link for phones to upload videos + + if request.data and request.data.get('fileName'): + fileName = '-' + request.data.get('fileName') # for result uploading - matching old way + else: # default: link for phones to upload videos fileName = '.mov' - - key = str(uuid.uuid4()) + fileName - + + key = str(uuid.uuid4()) + fileName + response = s3_client.generate_presigned_post( - Bucket = settings.AWS_STORAGE_BUCKET_NAME, - Key = key, - ExpiresIn = 1200 + Bucket=settings.AWS_STORAGE_BUCKET_NAME, + Key=key, + ExpiresIn=1200 ) return Response(response) - - ## Session status GET '/status/' - # if no active trial then return "ready" - # if there is an active trial then return "recording" - # if recording completed (trial set to "done") but some videos pending then "uploading" - # if recording and upload, but not processed then "processing" - # if "processing" returns errors or is done then go back to "ready" - # - # Logic on the client side: - # - if status changed "*" -> "recording" start recording - # - if status change "recording" -> "*" stop recording and submit the video - # - # For each device checking the status in the "recording" phase, create a video record + + # Session status GET '/status/' + @swagger_auto_schema( + operation_summary="Get Session Status", + responses={ + 200: openapi.Response("Success - Session status retrieved successfully."), + 404: openapi.Response("Not Found - Session not found."), + } + ) @action(detail=True) def status(self, request, pk): + """ + Retrieves the current status of a session. + + Statuses: + - "ready": No active trial, session is ready to start recording. + - "recording": An active trial is in progress and recording is ongoing. + - "uploading": Recording completed but some videos are still being uploaded. + - "processing": Videos are uploaded but still being processed. + - If "processing" results in errors or completes, the status reverts to "ready". + + Logic on the client side: + - if status changed "*" -> "recording" start recording + - if status change "recording" -> "*" stop recording and submit the video + + For each device checking the status in the "recording" phase, create a video record + """ status_dict = self.get_status(request, pk) - + return Response(status_dict) - ## Start recording POST '/record/' - # Create a new trial - # - creates a new trial with "recording" state + # Start recording POST '/record/' + @swagger_auto_schema( + operation_summary="Start a New Recording Session", + responses={ + 200: openapi.Response("Success - Recording session started successfully."), + 404: openapi.Response("Not Found - Session not found."), + 500: openapi.Response("Internal Server Error - Could not start recording session."), + } + ) @action(detail=True) def record(self, request, pk): + """ + Starts a new recording session by creating a trial. + + Creates a new trial associated with the given session and assigns it the status of "recording". + If the specified trial name already exists in the session, the function generates a new name + by appending a suffix. + """ def get_count_from_name(name, base_name): try: count = int(name[len(base_name) + 1:]) @@ -858,8 +1163,18 @@ def get_count_from_name(name, base_name): return Response(serializer.data) + @swagger_auto_schema( + operation_summary="Download Session Files as a ZIP", + responses={ + 200: openapi.Response("Success - File download initiated successfully."), + 500: openapi.Response("Internal Server Error - Could not initiate file download."), + } + ) @action(detail=True) def download(self, request, pk): + """ + Downloads the files associated with a session and returns them as a ZIP archive. + """ try: # Extract protocol and host. if request.is_secure(): @@ -876,11 +1191,7 @@ def download(self, request, pk): return FileResponse(open(session_zip, "rb")) - @action( - detail=True, - url_path="async-download", - url_name="async_session_download" - ) + @action(detail=True, url_path="async-download", url_name="async_session_download") def async_download(self, request, pk): try: if pk == 'undefined': @@ -901,9 +1212,24 @@ def async_download(self, request, pk): raise APIException(_('session_download_error')) return Response({"task_id": task.id}, status=200) - + + @swagger_auto_schema( + operation_summary="Get Session Permissions", + responses={ + 200: openapi.Response("Success - Session permissions retrieved successfully."), + 400: openapi.Response("Bad Request - Invalid trial data."), + 404: openapi.Response("Not Found - Trial not found."), + 500: openapi.Response("Internal Server Error - Could not retrieve session permissions."), + } + ) @action(detail=True) def get_session_permission(self, request, pk): + """ + Retrieves the permission settings for a specified session. + + This function checks the given session's ownership, visibility (public or private), + and whether the requesting user has admin privileges. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -932,8 +1258,22 @@ def get_session_permission(self, request, pk): return Response(sessionPermission) + @swagger_auto_schema( + operation_summary="Get Session Settings", + responses={ + 200: openapi.Response("Success - Session settings retrieved successfully."), + 400: openapi.Response("Bad Request - Invalid session data."), + 401: openapi.Response("Unauthorized - User must be authenticated."), + 403: openapi.Response("Forbidden - Authentication is required."), + 404: openapi.Response("Not Found - Session not found."), + 500: openapi.Response("Internal Server Error - Could not retrieve session settings."), + } + ) @action(detail=True) def get_session_settings(self, request, pk): + """ + Retrieves the settings of a specified session, including available framerate options. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -941,7 +1281,8 @@ def get_session_settings(self, request, pk): session = get_object_or_404(Session, pk=pk) # Check if using same setup - if session.meta and 'sessionWithCalibration' in session.meta and 'id' in session.meta['sessionWithCalibration']: + if session.meta and 'sessionWithCalibration' in session.meta and 'id' in session.meta[ + 'sessionWithCalibration']: session = Session.objects.get(pk=session.meta['sessionWithCalibration']['id']) self.check_object_permissions(self.request, session) @@ -990,44 +1331,170 @@ def get_session_settings(self, request, pk): return Response(settings_dict) + @swagger_auto_schema( + operation_summary="Set Session Metadata", + manual_parameters=[ + openapi.Parameter( + name="subject_id", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="ID of the subject for this session." + ), + openapi.Parameter( + name="subject_mass", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Mass of the subject." + ), + openapi.Parameter( + name="subject_height", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Height of the subject." + ), + openapi.Parameter( + name="subject_sex", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Sex of the subject." + ), + openapi.Parameter( + name="subject_gender", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Gender of the subject." + ), + openapi.Parameter( + name="subject_data_sharing", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Data sharing preferences for the subject." + ), + openapi.Parameter( + name="subject_pose_model", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Pose model used for the subject." + ), + openapi.Parameter( + name="settings_framerate", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Framerate setting for the session." + ), + openapi.Parameter( + name="settings_data_sharing", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Data sharing setting for the session." + ), + openapi.Parameter( + name="settings_pose_model", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Pose model setting for the session." + ), + openapi.Parameter( + name="settings_openSimModel", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="OpenSim model used for the session." + ), + openapi.Parameter( + name="settings_augmenter_model", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Augmenter model used for the session." + ), + openapi.Parameter( + name="settings_filter_frequency", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Filter frequency setting for the session." + ), + openapi.Parameter( + name="settings_scaling_setup", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Scaling setup used for the session." + ), + openapi.Parameter( + name="cb_square", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Size of each square in the checkerboard pattern." + ), + openapi.Parameter( + name="cb_rows", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Number of rows in the checkerboard pattern." + ), + openapi.Parameter( + name="cb_cols", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Number of columns in the checkerboard pattern." + ), + openapi.Parameter( + name="cb_placement", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Placement of the checkerboard." + ), + openapi.Parameter( + name="settings_session_name", + in_=openapi.IN_QUERY, + type=openapi.TYPE_STRING, + description="Name of the session." + ), + ], + responses={ + 200: openapi.Response("Success - Session metadata updated successfully."), + 400: openapi.Response("Bad Request - Invalid request data."), + 404: openapi.Response("Not Found - Session not found."), + 500: openapi.Response("Internal Server Error - Could not set the session metadata."), + } + ) @action(detail=True) def set_metadata(self, request, pk): - + """ + Updates the metadata of a specified session with provided query parameters. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) session = get_object_or_404(Session, pk=pk) - if not session.meta: session.meta = {} - + if "subject_id" in request.GET: session.meta["subject"] = { - "id": request.GET.get("subject_id",""), - "mass": request.GET.get("subject_mass",""), - "height": request.GET.get("subject_height",""), - "sex": request.GET.get("subject_sex",""), - "gender": request.GET.get("subject_gender",""), - "datasharing": request.GET.get("subject_data_sharing",""), - "posemodel": request.GET.get("subject_pose_model",""), + "id": request.GET.get("subject_id", ""), + "mass": request.GET.get("subject_mass", ""), + "height": request.GET.get("subject_height", ""), + "sex": request.GET.get("subject_sex", ""), + "gender": request.GET.get("subject_gender", ""), + "datasharing": request.GET.get("subject_data_sharing", ""), + "posemodel": request.GET.get("subject_pose_model", ""), } if "settings_framerate" in request.GET: session.meta["settings"] = { - "framerate": request.GET.get("settings_framerate",""), + "framerate": request.GET.get("settings_framerate", ""), } if "settings_data_sharing" in request.GET: if not session.meta["settings"]: session.meta["settings"] = {} - session.meta["settings"]["datasharing"] = request.GET.get("settings_data_sharing","") + session.meta["settings"]["datasharing"] = request.GET.get("settings_data_sharing", "") if "settings_pose_model" in request.GET: if not session.meta["settings"]: session.meta["settings"] = {} - session.meta["settings"]["posemodel"] = request.GET.get("settings_pose_model","") + session.meta["settings"]["posemodel"] = request.GET.get("settings_pose_model", "") if "settings_openSimModel" in request.GET: if not session.meta["settings"]: @@ -1048,13 +1515,13 @@ def set_metadata(self, request, pk): if not session.meta["settings"]: session.meta["settings"] = {} session.meta["settings"]["scalingsetup"] = request.GET.get("settings_scaling_setup", "") - + if "cb_square" in request.GET: session.meta["checkerboard"] = { - "square_size": request.GET.get("cb_square",""), - "rows": request.GET.get("cb_rows",""), - "cols": request.GET.get("cb_cols",""), - "placement": request.GET.get("cb_placement",""), + "square_size": request.GET.get("cb_square", ""), + "rows": request.GET.get("cb_rows", ""), + "cols": request.GET.get("cb_cols", ""), + "placement": request.GET.get("cb_placement", ""), } if "settings_session_name" in request.GET: @@ -1080,8 +1547,26 @@ def set_metadata(self, request, pk): return Response(serializer.data) + @swagger_auto_schema( + operation_summary="Assign a Subject to a Session", + manual_parameters=[ + openapi.Parameter( + 'subject_id', openapi.IN_QUERY, description="UUID of the subject to assign to the session", + type=openapi.TYPE_STRING, required=True + ), + ], + responses={ + 200: openapi.Response("Success - Session subject updated successfully."), + 400: openapi.Response("Bad Request - Invalid request data."), + 404: openapi.Response("Not Found - Session or subject not found."), + 500: openapi.Response("Internal Server Error - Could not update the subject."), + } + ) @action(detail=True) def set_subject(self, request, pk): + """ + Assign a Subject to a Session. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1123,13 +1608,24 @@ def set_subject(self, request, pk): return Response(serializer.data) - -## Stop recording POST '/stop/' - # Changes the trial status from "recording" to "done" - # Logic on the client side: - # - session status changed so they start uploading videos + # Stop recording POST '/stop/' + @swagger_auto_schema( + operation_summary="Stop Trial Recording", + responses={ + 200: openapi.Response("Success - Session stopped successfully."), + 400: openapi.Response("Bad Request - Invalid session data."), + 404: openapi.Response("Not Found - Session or trial not found."), + 500: openapi.Response("Internal Server Error - Could not stop the session."), + } + ) @action(detail=True) def stop(self, request, pk): + """ + Changes the trial status from "recording" to "done" + + Logic on the client side: + - Session status changed, so they start uploading videos. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1193,13 +1689,25 @@ def stop(self, request, pk): raise APIException(_('trial_cancel_error')) return Response(serializer.data) - - ## Cancel trial POST '/stop/' - # Changes the trial status from "stopped" to "error" - # Logic on the client side: - # - session status changed when cancel is pressed + + # Cancel trial POST '/stop/' + @swagger_auto_schema( + operation_summary="Cancel Trial", + responses={ + 200: openapi.Response("Success - Trial cancelled successfully."), + 400: openapi.Response("Bad Request - Invalid session data."), + 404: openapi.Response("Not Found - Session not found."), + 500: openapi.Response("Internal Server Error - Could not stop the trial."), + } + ) @action(detail=True) def cancel_trial(self, request, pk): + """ + Changes the trial status from "stopped" to "error" + + Logic on the client side: + - session status changed when cancel is pressed + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1230,8 +1738,21 @@ def cancel_trial(self, request, pk): return Response(data) + @swagger_auto_schema( + operation_summary="Retrieve Calibration Image and Status", + responses={ + 200: openapi.Response("Success - Calibration image retrieved successfully."), + 401: openapi.Response("Unauthorized - User must be authenticated."), + 403: openapi.Response("Forbidden - Authentication is required."), + 404: openapi.Response("Not Found - Session or trial not found."), + 500: openapi.Response("Internal Server Error - Could not retrieve calibration image."), + } + ) @action(detail=True) def calibration_img(self, request, pk): + """ + Retrieve Calibration Image and Status. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1252,7 +1773,7 @@ def calibration_img(self, request, pk): "n_cameras_connected": status_session["n_cameras_connected"], "n_videos_uploaded": status_session["n_videos_uploaded"] } - elif not trials[0].status in ['done', 'error']: # this gets updated on the backend by app.py + elif not trials[0].status in ['done', 'error']: # this gets updated on the backend by app.py data = { "status": "processing", "img": [ @@ -1267,7 +1788,7 @@ def calibration_img(self, request, pk): "img": "None", "n_cameras_connected": status_session["n_cameras_connected"], "n_videos_uploaded": status_session["n_videos_uploaded"] - } + } else: data = { @@ -1298,9 +1819,22 @@ def calibration_img(self, request, pk): raise APIException(_('calibration_image_retrieve_error')) return Response(data) - + + @swagger_auto_schema( + operation_summary="Retrieve Neutral Image and Status", + responses={ + 200: openapi.Response("Success - Neutral image retrieved successfully."), + 401: openapi.Response("Unauthorized - User must be authenticated."), + 403: openapi.Response("Forbidden - Authentication is required."), + 404: openapi.Response("Not Found - Session or trial not found."), + 500: openapi.Response("Internal Server Error - Could not retrieve neutral image."), + } + ) @action(detail=True) def neutral_img(self, request, pk): + """ + Retrieve Calibration Image and Status. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1320,7 +1854,7 @@ def neutral_img(self, request, pk): "n_cameras_connected": status_session["n_cameras_connected"], "n_videos_uploaded": status_session["n_videos_uploaded"] } - elif not trials[0].status in ['done', 'error']: # this gets updated on the backend by app.py + elif not trials[0].status in ['done', 'error']: # this gets updated on the backend by app.py data = { "status": "processing", "img": [ @@ -1343,12 +1877,12 @@ def neutral_img(self, request, pk): "n_videos_uploaded": status_session["n_videos_uploaded"] } else: - data = { + data = { "status": "error", "img": [ ], - "n_cameras_connected": status_session["n_cameras_connected"], - "n_videos_uploaded": status_session["n_videos_uploaded"] + "n_cameras_connected": status_session["n_cameras_connected"], + "n_videos_uploaded": status_session["n_videos_uploaded"] } except Http404: @@ -1374,9 +1908,41 @@ def neutral_img(self, request, pk): return Response(data) - + @swagger_auto_schema( + operation_summary="Retrieve Session Statuses", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + 'status': openapi.Schema( + type=openapi.TYPE_STRING, + description="Status of the sessions to filter (e.g., active, completed, error)." + ), + 'date_range': openapi.Schema( + type=openapi.TYPE_ARRAY, + items=openapi.Items(type=openapi.TYPE_STRING), + description="An array containing two date strings [start_date, end_date] for filtering sessions " + "by status change date." + ), + 'username': openapi.Schema( + type=openapi.TYPE_STRING, + description="Username of the user associated with the sessions (optional, requires admin or " + "backend permissions)." + ) + }, + required=['status'], + ), + responses={ + 200: openapi.Response("Success - Session statuses retrieved successfully."), + 400: openapi.Response("Bad Request - Invalid session data."), + 404: openapi.Response("Not Found - Session not found."), + 500: openapi.Response("Internal Server Error - Could not retrieve session statuses."), + } + ) @action(detail=False, methods=['post'], permission_classes=[IsAdmin | IsBackend | IsOwner]) def get_session_statuses(self, request): + """ + Retrieve Session Statuses. + """ from .serializers import SessionIdSerializer, SessionFilteringSerializer try: filtering_serializer = SessionFilteringSerializer(data=request.data) @@ -1399,11 +1965,11 @@ def get_session_statuses(self, request): except Http404: if settings.DEBUG: raise APIException(_("error") % {"error_message": str(traceback.format_exc())}) - raise NotFound(_("session_uuid_not_found") % {"uuid": str(pk)}) + raise NotFound(_("user_not_found")) except ValueError: if settings.DEBUG: raise APIException(_("error") % {"error_message": str(traceback.format_exc())}) - raise NotFound(_("session_uuid_not_valid") % {"uuid": str(pk)}) + raise NotFound(_("session_not_valid")) except Exception: if settings.DEBUG: raise APIException(_("error") % {"error_message": str(traceback.format_exc())}) @@ -1411,9 +1977,31 @@ def get_session_statuses(self, request): return Response(serializer.data) - + @swagger_auto_schema( + operation_summary="Set Session Status", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + 'status': openapi.Schema( + type=openapi.TYPE_STRING, + description="The new status to set for the session (e.g., active, completed, error).", + example="completed" + ) + }, + required=['status'], + ), + responses={ + 200: openapi.Response("Success - Session status set successfully."), + 400: openapi.Response("Bad Request - Invalid session data."), + 404: openapi.Response("Not Found - Session not found."), + 500: openapi.Response("Internal Server Error - Could not set session status."), + } + ) @action(detail=True, methods=['post'], permission_classes=[IsAdmin | IsBackend]) def set_session_status(self, request, pk): + """ + Set Session Status. + """ from .serializers import SessionStatusSerializer try: if pk == 'undefined': @@ -1442,19 +2030,32 @@ def set_session_status(self, request, pk): return Response(serializer.data) - ## Processing machine: # A worker asks whether there is any trial to process # - if no it asks again in 5 sec # - if yes it runs processing and sends back the results class TrialViewSet(viewsets.ModelViewSet): + """ + A view set for viewing and editing trials. + """ queryset = Trial.objects.all().order_by("created_at") serializer_class = TrialSerializer permission_classes = [IsPublic | (IsOwner | IsAdmin | IsBackend)] - - @action(detail=False, permission_classes=[((IsAdmin | IsBackend))]) + + @swagger_auto_schema( + operation_summary="Dequeue a trial for processing", + responses={ + 200: openapi.Response("Success - Trial dequeued successfully."), + 400: openapi.Response("Bad Request - Invalid trial data."), + 404: openapi.Response("Not Found - Trial not found."), + }, + ) + @action(detail=False, permission_classes=[(IsAdmin | IsBackend)]) def dequeue(self, request): + """ + Dequeue a trial and set its status to 'processing' if available. + """ try: ip = get_client_ip(request) @@ -1462,44 +2063,45 @@ def dequeue(self, request): # find trials with some videos not uploaded not_uploaded = Video.objects.filter(video='', - updated_at__gte=datetime.now() + timedelta(minutes=-15)).values_list("trial__id", flat=True) + updated_at__gte=datetime.now() + timedelta(minutes=-15)).values_list( + "trial__id", flat=True) print(not_uploaded) uploaded_trials = Trial.objects.exclude(id__in=not_uploaded) - # uploaded_trials = Trial.objects.all() + # uploaded_trials = Trial.objects.all() if workerType != 'dynamic': # Priority for 'calibration' and 'neutral' trials = uploaded_trials.filter(status="stopped", - name__in=["calibration","neutral"], - result=None) + name__in=["calibration", "neutral"], + result=None) trialsReprocess = uploaded_trials.filter(status="reprocess", - name__in=["calibration","neutral"], - result=None) + name__in=["calibration", "neutral"], + result=None) if trials.count() == 0 and workerType != 'calibration': trials = uploaded_trials.filter(status="stopped", - result=None) + result=None) - if trials.count()==0 and trialsReprocess.count() == 0 and workerType != 'calibration': + if trials.count() == 0 and trialsReprocess.count() == 0 and workerType != 'calibration': trialsReprocess = uploaded_trials.filter(status="reprocess", - result=None) + result=None) else: trials = uploaded_trials.filter(status="stopped", result=None).exclude(name__in=["calibration", "neutral"]) trialsReprocess = uploaded_trials.filter(status="reprocess", - result=None).exclude(name__in=["calibration", "neutral"]) - + result=None).exclude(name__in=["calibration", "neutral"]) if trials.count() == 0 and trialsReprocess.count() == 0: raise Http404 - # prioritize admin and priority group trials (priority group doesn't exist yet, but should have same priv. as user) - trialsPrioritized = trials.filter(session__user__groups__name__in=["admin","priority"]) + # prioritize admin and priority group trials (priority group doesn't exist yet, but should have same + # priv. as user) + trialsPrioritized = trials.filter(session__user__groups__name__in=["admin", "priority"]) # if not priority trials, go to normal trials if trialsPrioritized.count() == 0: trialsPrioritized = trials @@ -1530,27 +2132,51 @@ def dequeue(self, request): raise APIException(_('trial_dequeue_error')) return Response(serializer.data) - + + @swagger_auto_schema( + operation_summary="Get Trials by Status", + responses={ + 200: openapi.Response("Success - Trial list retrieved successfully."), + 400: openapi.Response("Bad Request - Invalid search data."), + }, + ) @action(detail=False, permission_classes=[((IsAdmin | IsBackend))]) def get_trials_with_status(self, request): """ - This view returns a list of all the trials with the specified status - that was updated more than hoursSinceUpdate hours ago. + Returns a list of trials with the specified status that were updated more than 'hoursSinceUpdate' hours ago. """ hours_since_update = request.query_params.get('hoursSinceUpdate', 0) - hours_since_update = float(hours_since_update) if hours_since_update else 0 + hours_since_update = float(hours_since_update) if hours_since_update else 0 status = self.request.query_params.get('status') # trials with given status and updated_at more than n hours ago trials = Trial.objects.filter(status=status, - updated_at__lte=(datetime.now() - timedelta(hours=hours_since_update))).order_by("-created_at") - + updated_at__lte=(datetime.now() - timedelta(hours=hours_since_update))).order_by( + "-created_at") + serializer = TrialSerializer(trials, many=True) return Response(serializer.data) + @swagger_auto_schema( + operation_summary="Rename a specific trial by ID", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + 'trialNewName': openapi.Schema(type=openapi.TYPE_STRING, description='New name for the trial'), + } + ), + responses={ + 200: openapi.Response("Success - Trial renamed successfully."), + 400: openapi.Response("Bad Request - Invalid trial data."), + 404: openapi.Response("Not Found - Trial not found."), + }, + ) @action(detail=True, methods=['post']) def rename(self, request, pk): + """ + Rename a specific trial by its ID. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1583,8 +2209,19 @@ def rename(self, request, pk): 'data': serializer.data }) + @swagger_auto_schema( + operation_summary="Permanently Remove a Trial", + responses={ + 204: openapi.Response("Deleted - Trial deleted successfully."), + 400: openapi.Response("Bad Request - Invalid trial data."), + 404: openapi.Response("Not Found - Trial not found."), + }, + ) @action(detail=True, methods=['post']) def permanent_remove(self, request, pk): + """ + Permanently delete a trial by its ID. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1607,8 +2244,24 @@ def permanent_remove(self, request, pk): return Response({}) + @swagger_auto_schema( + operation_summary="Trash trial", + operation_description="Move a trial to the trash.", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={"pk": openapi.Schema(type=openapi.TYPE_STRING)}, + required=["pk"] + ), + responses={ + 200: openapi.Response("Succes - Trial trashed successfully."), + 404: openapi.Response("Not Found - Trial not found."), + } + ) @action(detail=True, methods=['post']) def trash(self, request, pk): + """ + Move a specific trial to the trash. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1635,8 +2288,19 @@ def trash(self, request, pk): return Response(serializer.data) + @swagger_auto_schema( + operation_summary="Restore a Trial from Trash", + responses={ + 200: openapi.Response("Success - Trial restored from trash successfully."), + 400: openapi.Response("Bad Request - Invalid trial data."), + 404: openapi.Response("Not Found - Trial not found."), + }, + ) @action(detail=True, methods=['post']) def restore(self, request, pk): + """ + Restore a specific trial from the trash. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1663,6 +2327,88 @@ def restore(self, request, pk): return Response(serializer.data) + @swagger_auto_schema( + operation_summary="List Trials", + responses={ + 200: openapi.Response("Success - List of trials retrieved successfully."), + }, + ) + def list(self, request, *args, **kwargs): + """ + Retrieve a list of trials. + """ + return super().list(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Create a Trial", + request_body=VideoSerializer, + responses={ + 201: openapi.Response("Created - Trial created successfully."), + 404: openapi.Response("Not Found - Trial not found."), + 403: openapi.Response("Forbidden - Authentication is required."), + }, + ) + def create(self, request, *args, **kwargs): + """ + Create a new trial instance. + """ + return super().create(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Retrieve Trial", + responses={ + 200: openapi.Response("Success - Trial retrieved successfully."), + 404: openapi.Response("Not Found - Trial not found."), + }, ) + def retrieve(self, request, *args, **kwargs): + """ + Retrieve a specific trial instance by ID. + """ + return super().retrieve(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Update Trial", + request_body=VideoSerializer, + responses={ + 200: openapi.Response("Success - Trial updated successfully."), + 400: openapi.Response("Bad Request - Invalid trial data."), + 404: openapi.Response("Not Found - Trial not found."), + }, + ) + def update(self, request, *args, **kwargs): + """ + Update a specific trial instance by ID. + """ + return super().update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Partial Update Trial", + request_body=VideoSerializer, + responses={ + 200: openapi.Response("Success - Trial partially updated successfully."), + 400: openapi.Response("Bad Request - Invalid trial data."), + 404: openapi.Response("Not Found - Trial not found."), + }, + ) + def partial_update(self, request, *args, **kwargs): + """ + Partially update a specific trial instance by ID. + """ + return super().partial_update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Delete Trial", + responses={ + 204: openapi.Response("Deleted - Trial deleted successfully."), + 404: openapi.Response("Not Found - Trial not found."), + }, + ) + def destroy(self, request, *args, **kwargs): + """ + Delete a specific trial instance by ID. + """ + return super().destroy(request, *args, **kwargs) + @action(detail=True, methods=['post']) def modifyTags(self, request, pk): try: @@ -1701,19 +2447,19 @@ def modifyTags(self, request, pk): 'data': serializer.data }) - - - ## Upload a video: # Input: video and phone_id # Logic: Find the Video model within this session with # device_id. Upload Video to that model class VideoViewSet(viewsets.ModelViewSet): + """ + A view set for viewing and editing videos. + """ queryset = Video.objects.all().order_by("-created_at") serializer_class = VideoSerializer permission_classes = [AllowPublicCreate | ((IsOwner | IsAdmin | IsBackend))] - + def perform_update(self, serializer): if ("video_url" in serializer.validated_data) and (serializer.validated_data["video_url"]): serializer.validated_data["video"] = serializer.validated_data["video_url"] @@ -1721,13 +2467,112 @@ def perform_update(self, serializer): super().perform_update(serializer) + @swagger_auto_schema( + operation_summary="List Videos", + responses={ + 200: openapi.Response("Success - List of videos retrieved successfully."), + }, + ) + def list(self, request, *args, **kwargs): + """ + Retrieve a list of videos. + """ + return super().list(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Create Video", + request_body=VideoSerializer, + responses={ + 201: openapi.Response("Created - Video created successfully."), + 404: openapi.Response("Not Found - Video not found."), + 403: openapi.Response("Forbidden - Authentication is required."), + }, + ) + def create(self, request, *args, **kwargs): + """ + Create a new video instance. + """ + return super().create(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Retrieve Video", + responses={ + 200: openapi.Response("Success - Video retrieved successfully."), + 404: openapi.Response("Not Found - Video not found."), + }, + ) + def retrieve(self, request, *args, **kwargs): + """ + Retrieve a specific video instance by ID. + """ + return super().retrieve(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Update Video", + request_body=VideoSerializer, + responses={ + 200: openapi.Response("Success - Video updated successfully."), + 400: openapi.Response("Bad Request - Invalid video data."), + 404: openapi.Response("Not Found - Video not found."), + }, + ) + def update(self, request, *args, **kwargs): + """ + Update a specific video instance by ID. + """ + return super().update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Partial Update Video", + request_body=VideoSerializer, + responses={ + 200: openapi.Response("Success - Video partially updated successfully."), + 400: openapi.Response("Bad Request - Invalid video data."), + 404: openapi.Response("Not Found - Video not found."), + }, + ) + def partial_update(self, request, *args, **kwargs): + """ + Partially update a specific video instance by ID. + """ + return super().partial_update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Delete Video", + responses={ + 204: openapi.Response("Deleted - Video deleted successfully."), + 404: openapi.Response("Not Found - Video not found."), + }, + ) + def destroy(self, request, *args, **kwargs): + """ + Delete a specific video instance by ID. + """ + return super().destroy(request, *args, **kwargs) + + class ResultViewSet(viewsets.ModelViewSet): + """ + A view set for viewing and editing results. + """ queryset = Result.objects.all().order_by("-created_at") serializer_class = ResultSerializer permission_classes = [IsOwner | IsAdmin | IsBackend] + @swagger_auto_schema( + operation_summary="Create Result", + request_body=ResultSerializer, + responses={ + 201: openapi.Response("Created - Result created successfully."), + 400: openapi.Response("Bad Request - Invalid Result data."), + 403: openapi.Response("Forbidden - Authentication is required."), + }, + ) def create(self, request): + """ + Create a new result instance. + """ serializer = self.get_serializer(data=request.data) serializer.is_valid(raise_exception=True) @@ -1745,23 +2590,114 @@ def create(self, request): headers = self.get_success_headers(serializer.data) return Response(serializer.data, status=status.HTTP_201_CREATED, headers=headers) + @swagger_auto_schema( + operation_summary="List Results", + responses={ + 200: openapi.Response("Success - List of results retrieved successfully."), + }, + ) + def list(self, request, *args, **kwargs): + """ + Retrieve a list of results. + """ + return super().list(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Retrieve Result", + responses={ + 200: openapi.Response("Success - Rrsult retrieved successfully."), + 404: openapi.Response("Not Found - Result not found."), + }, + ) + def retrieve(self, request, *args, **kwargs): + """ + Retrieve a specific result instance by ID. + """ + return super().retrieve(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Update Result", + request_body=ResultSerializer, + responses={ + 200: openapi.Response("Success - Result updated successfully."), + 400: openapi.Response("Bad Request - Invalid Result data."), + 404: openapi.Response("Not Found - Result not found."), + }, + ) + def update(self, request, *args, **kwargs): + """ + Update a specific result instance by ID. + """ + return super().update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Partial Update Result", + request_body=ResultSerializer, + responses={ + 200: openapi.Response("Success - Result partially updated successfully."), + 400: openapi.Response("Bad Request - Invalid result data."), + 404: openapi.Response("Not Found - Result not found."), + }, + ) + def partial_update(self, request, *args, **kwargs): + """ + Partially update a specific result instance by ID. + """ + return super().partial_update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Delete Result", + responses={ + 204: openapi.Response("Deleted - Result deleted successfully."), + 404: openapi.Response("Not Found - Result not found."), + }, + ) + def destroy(self, request, *args, **kwargs): + """ + Delete a specific result instance by ID. + """ + return super().destroy(request, *args, **kwargs) + class SubjectViewSet(viewsets.ModelViewSet): + """ + A view set for viewing and editing subjects. + """ permission_classes = [IsOwner | IsAdmin | IsBackend] def get_queryset(self): """ - This view should return a list of all the subjects - for the currently authenticated user. + Returns a list of subjects for the currently authenticated user. + + - Admin users get all subjects. + - Regular users get only their own subjects. """ user = self.request.user + # If user is authenticated, and it has privileges (e.g., is admin) return all. if (user.is_authenticated and user.id == 1) or (user.is_authenticated and user.id == 2): return Subject.objects.all().prefetch_related('subjecttags_set') + # If user is authenticated, but has no privileges, return only its own data. + elif user.is_authenticated and type(user) is not AnonymousUser: + return Subject.objects.filter(user=user).prefetch_related('subjecttags_set') + else: + return [] # public_subject_ids = Session.objects.filter(public=True).values_list('subject_id', flat=True).distinct() # return Subject.objects.filter(Q(user=user) | Q(id__in=public_subject_ids)).prefetch_related('subjecttags_set') - return Subject.objects.filter(user=user).prefetch_related('subjecttags_set') - + # return Subject.objects.filter(user=user).prefetch_related('subjecttags_set') + + @swagger_auto_schema( + operation_summary="List subjects", + responses={ + 200: openapi.Response("Success - List of subjects retrieved successfully."), + 400: openapi.Response("Bad Request - Invalid search parameters."), + 401: openapi.Response("Unauthorized - User must be authenticated."), + } + ) def list(self, request): + """ + Retrieves a list of subjects based on the user's permissions and provided query parameters. + The list can be filtered, sorted, and paginated using the provided options. + """ queryset = self.get_queryset() # Get quantity from post request. If it does exist, use it. If not, set -1 as default (e.g., return all) # print(request.query_params) @@ -1804,12 +2740,37 @@ def get_serializer_class(self): return NewSubjectSerializer return SubjectSerializer + @swagger_auto_schema( + operation_summary="Health check", + responses={ + 200: openapi.Response("Success - API health retrieved successfully."), + } + ) @action(detail=False) def api_health_check(self, request): + """ + Check the health of the API. + """ return Response({"status": "True"}) + @swagger_auto_schema( + operation_summary="Trash subject", + operation_description="Move a subject to the trash.", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={"pk": openapi.Schema(type=openapi.TYPE_STRING)}, + required=["pk"] + ), + responses={ + 200: openapi.Response("Success - Subject trashed successfully."), + 404: openapi.Response("Not Found - Subject not found."), + } + ) @action(detail=True, methods=['post']) def trash(self, request, pk): + """ + Move a specific subject to the trash. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1836,8 +2797,24 @@ def trash(self, request, pk): return Response(serializer.data) + @swagger_auto_schema( + operation_summary="Restore subject", + operation_description="Restore a subject from the trash.", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={"pk": openapi.Schema(type=openapi.TYPE_STRING)}, + required=["pk"] + ), + responses={ + 200: openapi.Response("Success - Subject restored successfully."), + 404: openapi.Response("Not Found - Subject not found."), + } + ) @action(detail=True, methods=['post']) def restore(self, request, pk): + """ + Restores a previously trashed subject. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1863,8 +2840,18 @@ def restore(self, request, pk): return Response(serializer.data) + @swagger_auto_schema( + operation_summary="Download subject", + responses={ + 200: openapi.Response("Success - Subject archive downloaded successfully."), + 404: openapi.Response("Not Found - Subject not found."), + } + ) @action(detail=True) def download(self, request, pk): + """ + Downloads a zip file containing the data for the specified subject. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1891,13 +2878,19 @@ def download(self, request, pk): raise APIException(_('subject_create_error')) return FileResponse(open(subject_zip, "rb")) - - @action( - detail=True, - url_path="async-download", - url_name="async_subject_download" + + @swagger_auto_schema( + operation_summary="Async download", + responses={ + 200: openapi.Response("Success - Subject archive downloaded successfully."), + 404: openapi.Response("Not Found - Subject not found."), + } ) + @action(detail=True, url_path="async-download", url_name="async_subject_download") def async_download(self, request, pk): + """ + Download a subject archive. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1920,8 +2913,23 @@ def async_download(self, request, pk): return Response({"task_id": task.id}, status=200) + @swagger_auto_schema( + operation_summary="Permanently remove subject", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={"pk": openapi.Schema(type=openapi.TYPE_STRING)}, + required=["pk"] + ), + responses={ + 200: openapi.Response("Success - Subject removed successfully."), + 404: openapi.Response("Not Found - Subject not found."), + } + ) @action(detail=True, methods=['post']) def permanent_remove(self, request, pk): + """ + Permanently deletes a subject. + """ try: if pk == 'undefined': raise ValueError(_("undefined_uuid")) @@ -1972,25 +2980,107 @@ def perform_update(self, serializer): raise APIException(_("error") % {"error_message": str(traceback.format_exc())}) raise APIException(_('subject_update_error')) + @swagger_auto_schema( + operation_summary="Retrieve Subject", + responses={ + 200: openapi.Response("Success - Subject retrieved successfully."), + 404: openapi.Response("Not Found - Subject not found."), + }, + ) + def retrieve(self, request, *args, **kwargs): + """ + Retrieve a specific subject instance by ID. + """ + return super().retrieve(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Update Subject", + request_body=VideoSerializer, + responses={ + 200: openapi.Response("Success - Subject updated successfully."), + 400: openapi.Response("Bad Request - Invalid subject data."), + 404: openapi.Response("Not Found - Subject not found."), + }, + ) + def update(self, request, *args, **kwargs): + """ + Update a specific subject instance by ID. + """ + return super().update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Partial Update Subject", + request_body=VideoSerializer, + responses={ + 200: openapi.Response("Success - Subject partially updated successfully."), + 400: openapi.Response("Bad Request - Invalid Subject data."), + 404: openapi.Response("Not Found - Subject not found."), + }, + ) + def partial_update(self, request, *args, **kwargs): + """ + Partially update a specific subject instance by ID. + """ + return super().partial_update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Delete Subject", + responses={ + 204: openapi.Response("Deleted - Subject deleted successfully."), + 404: openapi.Response("Not Found - Subject not found."), + }, + ) + def destroy(self, request, *args, **kwargs): + """ + Delete a specific subject instance by ID. + """ + return super().destroy(request, *args, **kwargs) + + class SubjectTagViewSet(viewsets.ModelViewSet): + """ + A view set for viewing and editing subject tags. + """ permission_classes = [IsOwner | IsAdmin | IsBackend] serializer_class = SubjectTagSerializer def get_queryset(self): """ - This view should return a list of all the subjects tags - for the currently authenticated user. + Retrieve a list of all the subject tags for the currently authenticated user. + + - If the user is authenticated, returns the tags associated with their subjects. + - If the user is not authenticated, returns an empty list. """ - # Get all subjects associated to a user. - subject = Subject.objects.filter(user=self.request.user) + user = self.request.user + if user.is_authenticated: + # Get all subjects associated to a user. + subject = Subject.objects.filter(user=self.request.user) - # Get tags associated to those subjects. - tags = SubjectTags.objects.filter(subject__in=list(subject)) + # Get tags associated to those subjects. + tags = SubjectTags.objects.filter(subject__in=list(subject)) + else: + tags = [] return tags + @swagger_auto_schema( + operation_summary="Get tags for a specific subject", + operation_description="Retrieve tags associated with a specific subject identified by its ID.", + manual_parameters=[ + openapi.Parameter('subject_id', openapi.IN_PATH, description="ID of the subject to retrieve tags for.", + type=openapi.TYPE_INTEGER), + ], + responses={ + 200: openapi.Response("Success - Subject tags retrieved successfully."), + 403: openapi.Response("Forbidden - Authentication is required."), + 404: openapi.Response("Not Found - Subject tags not found."), + } + ) @action(detail=False, methods=['get']) def get_tags_subject(self, request, subject_id): + """ + Retrieves the tags associated with a specific subject. + """ # Get subject associated to that id. subject = Subject.objects.filter(id=subject_id).first() @@ -2000,9 +3090,92 @@ def get_tags_subject(self, request, subject_id): return Response(tags, status=200) else: - return Response(_("Subject with id: ") + str(subject_id) + _(" does not exist for user ") + self.request.user.username, status=404) - - + return Response( + _("Subject with id: ") + str(subject_id) + _(" does not exist for user ") + self.request.user.username, + status=404) + + @swagger_auto_schema( + operation_summary="List Subject Tags", + responses={ + 200: openapi.Response("Success - List of subject tags retrieved successfully."), + }, + ) + def list(self, request, *args, **kwargs): + """ + Retrieve a list of subject tags. + """ + return super().list(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Create Subject Tag", + request_body=UserSerializer, + responses={ + 201: openapi.Response("Created - Subject tag created successfully."), + 400: openapi.Response("Bad Request - Invalid subject tag data."), + 403: openapi.Response("Forbidden - Authentication is required."), + }, + ) + def create(self, request, *args, **kwargs): + """ + Create a new subject tag instance. + """ + return super().create(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Retrieve Subject Tag", + responses={ + 200: openapi.Response("Success - Subject tag retrieved successfully."), + 404: openapi.Response("Not Found - Subject tag not found."), + }, + ) + def retrieve(self, request, *args, **kwargs): + """ + Retrieve a specific subject tag instance by ID. + """ + return super().retrieve(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Update Subject Tag", + request_body=VideoSerializer, + responses={ + 200: openapi.Response("Success - Subject tag updated successfully."), + 400: openapi.Response("Bad Request - Invalid subject tag data."), + 404: openapi.Response("Not Found - Subject tag not found."), + }, + ) + def update(self, request, *args, **kwargs): + """ + Update a specific subject tag instance by ID. + """ + return super().update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Partial Update Subject Tag", + request_body=VideoSerializer, + responses={ + 200: openapi.Response("Success - Subject tag partially updated successfully."), + 400: openapi.Response("Bad Request - Invalid subject tag data."), + 404: openapi.Response("Not Found - Subject tag not found."), + }, + ) + def partial_update(self, request, *args, **kwargs): + """ + Partially update a specific subject tag instance by ID. + """ + return super().partial_update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Delete Subject Tag", + responses={ + 204: openapi.Response("Deleted - Subject tag deleted successfully."), + 404: openapi.Response("Not Found - Subject tag not found."), + }, + ) + def destroy(self, request, *args, **kwargs): + """ + Delete a specific subject tag instance by ID. + """ + return super().destroy(request, *args, **kwargs) class TrialTagViewSet(viewsets.ModelViewSet): permission_classes = [IsOwner | IsAdmin | IsBackend] @@ -2038,9 +3211,23 @@ def get_tags_trial(self, request, trial_id): class DownloadFileOnReadyAPIView(APIView): + """ + Retrieves the download URL for a file if it is ready. + If the file is not ready, returns a 202 status to indicate processing. + """ permission_classes = (AllowAny,) + @swagger_auto_schema( + operation_summary="Get Download URL", + responses={ + 200: openapi.Response("Success - File URL for download retrieved successfully."), + 202: openapi.Response("Accepted - The file was accepted and its being processed."), + } + ) def get(self, request, *args, **kwargs): + """ + Check if the download file is ready. + """ log = DownloadLog.objects.filter(task_id=self.kwargs["task_id"]).first() if log and log.media: return Response({"url": log.media.url}) @@ -2048,19 +3235,127 @@ def get(self, request, *args, **kwargs): class UserViewSet(viewsets.ModelViewSet): + """ + A view set for viewing and editing users. + """ queryset = User.objects.all() serializer_class = UserSerializer permission_classes = [IsAdmin] - - + + @swagger_auto_schema( + operation_summary="List Users", + responses={ + 200: openapi.Response("Success - User retrieved successfully."), + }, + ) + def list(self, request, *args, **kwargs): + """ + Retrieve a list of users. + """ + return super().list(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Create User", + request_body=UserSerializer, + responses={ + 201: openapi.Response("Created - User created successfully."), + 400: openapi.Response("Bad Request - Invalid user data."), + 403: openapi.Response("Forbidden - Authentication is required."), + }, + ) + def create(self, request, *args, **kwargs): + """ + Create a new user instance. + """ + return super().create(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Retrieve User", + responses={ + 200: openapi.Response("Success - User retrieved successfully."), + 404: openapi.Response("Not Found - User not found."), + }, + ) + def retrieve(self, request, *args, **kwargs): + """ + Retrieve a specific user instance by ID. + """ + return super().retrieve(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Update User", + request_body=UserSerializer, + responses={ + 200: openapi.Response("Success - User updated successfully."), + 400: openapi.Response("Bad Request - Invalid user data."), + 404: openapi.Response("Not Found - User not found."), + }, + ) + def update(self, request, *args, **kwargs): + """ + Update a specific user instance by ID. + """ + return super().update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Partial Update User", + request_body=UserSerializer, + responses={ + 200: openapi.Response("Success - User partially updated successfully."), + 400: openapi.Response("Bad Request - Invalid user data."), + 404: openapi.Response("Not Found - User not found."), + }, + ) + def partial_update(self, request, *args, **kwargs): + """ + Partially update a specific user instance by ID. + """ + return super().partial_update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Delete User", + responses={ + 204: openapi.Response("Deleted - User deleted successfully."), + 404: openapi.Response("Not Found - User not found."), + + }, + ) + def destroy(self, request, *args, **kwargs): + """ + Delete a specific user instance by ID. + """ + return super().destroy(request, *args, **kwargs) + + class UserCreate(APIView): """ - Creates the user. + Creates a new user and returns the user data along with an authentication token. """ permission_classes = [AllowAny] + @swagger_auto_schema( + operation_summary="Create User", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + 'username': openapi.Schema(type=openapi.TYPE_STRING, description='Username of the user'), + 'email': openapi.Schema(type=openapi.TYPE_STRING, description='Email of the user'), + 'password': openapi.Schema(type=openapi.TYPE_STRING, description='Password for the user'), + # Add additional fields as necessary based on your UserSerializer + }, + required=['username', 'email', 'password'], + ), + responses={ + 201: openapi.Response("Created - User created successfully."), + 400: openapi.Response("Bad Request - Invalid user information."), + 500: openapi.Response("Internal Server Error - Could not create user.") + } + ) def post(self, request, format='json'): + """ + Handles the POST request to create a new user. + """ try: serializer = UserSerializer(data=request.data) if serializer.is_valid(): @@ -2077,13 +3372,32 @@ def post(self, request, format='json'): return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) + class UserDelete(APIView): """ - Deletes the user. + Deletes a user. Requires confirmation by providing the username in the request data. """ permission_classes = [IsAuthenticated] + @swagger_auto_schema( + operation_summary="Delete User Account", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + 'confirm': openapi.Schema(type=openapi.TYPE_STRING, description="Username for confirmation"), + }, + required=['confirm'], + ), + responses={ + 200: openapi.Response("Success - User deleted successfully."), + 400: openapi.Response("Bad Request - Invalid username confirmation."), + 401: openapi.Response("Unauthorized - User must be authenticated.") + } + ) def post(self, request, format='json'): + """ + Handle POST requests to delete the user account. + """ try: if "confirm" not in request.data: return Response(_('user_delete_error'), status=status.HTTP_400_BAD_REQUEST) @@ -2103,13 +3417,27 @@ def post(self, request, format='json'): raise APIException(_("error") % {"error_message": str(traceback.format_exc())}) raise APIException(_('user_delete_error')) + class UserUpdate(APIView): """ - Updates the user. + Updates a user. """ permission_classes = [IsAuthenticated] + @swagger_auto_schema( + operation_summary="Update User", + request_body=UserUpdateSerializer, + responses={ + 200: openapi.Response("Success - User information updated successfully."), + 400: openapi.Response("Bad Request - Invalid user information."), + 401: openapi.Response("Unauthorized - User must be authenticated."), + 500: openapi.Response("Internal Server Error - Could not update user information.") + }, + ) def post(self, request, format='json'): + """ + Updates the authenticated user's information. + """ try: user = request.user serializer = UserUpdateSerializer(user, data=request.data, partial=True) @@ -2130,7 +3458,20 @@ class UpdateProfilePicture(APIView): """ permission_classes = [IsAuthenticated] + @swagger_auto_schema( + operation_summary="Update Profile Picture", + request_body=ProfilePictureSerializer, + responses={ + 200: openapi.Response("Success - Profile picture updated successfully."), + 400: openapi.Response("Bad Request - Invalid profile picture."), + 401: openapi.Response("Unauthorized - User must be authenticated."), + 500: openapi.Response("Internal Server Error - Could not update profile picture.") + }, + ) def post(self, request, format='json'): + """ + Updates the profile picture for the authenticated user. + """ try: user = request.user serializer = ProfilePictureSerializer(user, data=request.data, partial=True) @@ -2144,13 +3485,32 @@ def post(self, request, format='json'): return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST) + class GetUserInfo(APIView): """ - Retrieves info about a user. + Retrieves information about a user based on a username. """ permission_classes = [AllowAny] + @swagger_auto_schema( + operation_summary="Get User Info", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + 'username': openapi.Schema(type=openapi.TYPE_STRING, + description="The username of the user to retrieve information for."), + }, + required=['username'], + ), + responses={ + 200: openapi.Response("Success - User information retrieved successfully."), + 404: openapi.Response("Not Found - The requested user does not exist."), + } + ) def post(self, request, format='json'): + """ + Handle POST requests to retrieve user information based on username. + """ username = request.data["username"] user = get_object_or_404(User, username__exact=username) @@ -2176,8 +3536,29 @@ def post(self, request, format='json'): class CustomAuthToken(ObtainAuthToken): + """ + Custom authentication token view that handles user login and OTP verification. + """ + @swagger_auto_schema( + operation_summary="Obtain Auth Token", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + 'username': openapi.Schema(type=openapi.TYPE_STRING, description="Username of the user."), + 'password': openapi.Schema(type=openapi.TYPE_STRING, description="Password of the user."), + }, + required=['username', 'password'], + ), + responses={ + 200: openapi.Response("Success - Logged in successfully."), + 400: openapi.Response("Bad Request - Invalid credentials or other login error."), + } + ) def post(self, request, *args, **kwargs): + """ + Handles POST requests for user authentication and OTP verification. + """ try: serializer = self.serializer_class(data=request.data, context={'request': request}) @@ -2190,7 +3571,7 @@ def post(self, request, *args, **kwargs): # Skip OTP verification if specified otp_challenge_sent = False - if not(user.otp_verified and user.otp_skip_till and user.otp_skip_till > timezone.now()): + if not (user.otp_verified and user.otp_skip_till and user.otp_skip_till > timezone.now()): user.otp_verified = False user.save() @@ -2217,15 +3598,41 @@ def post(self, request, *args, **kwargs): 'institutional_use': user.institutional_use, }) + class ResetPasswordView(APIView): + """ + Initiates the password reset process by sending a reset email. + """ permission_classes = [AllowAny] authentication_classes = [] + @swagger_auto_schema( + operation_summary="Reset Password", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + 'email': openapi.Schema(type=openapi.TYPE_STRING, + description='Email of the user requesting a password reset'), + 'host': openapi.Schema(type=openapi.TYPE_STRING, description='Host URL for generating the reset link'), + }, + required=['email', 'host'], + ), + responses={ + 200: openapi.Response("Success - Email sent successfully."), + 404: openapi.Response("Not Found - Email not found."), + 500: openapi.Response("Internal Server Error - Could not initiate password reset.") + } + ) def post(self, request, format='json'): + """ + Handles the POST request to send a password reset email. + + Validates the input data and sends a reset password email if valid. + """ try: error_message = "success" serializer = ResetPasswordSerializer(data=request.data, - context={'request': request}) + context={'request': request}) serializer.is_valid(raise_exception=True) email = serializer.validated_data['email'] @@ -2272,9 +3679,32 @@ def post(self, request, format='json'): class NewPasswordView(APIView): + """ + Allows users to set a new password using a reset token. + """ permission_classes = [AllowAny] + @swagger_auto_schema( + operation_summary="Reset Password", + request_body=openapi.Schema( + type=openapi.TYPE_OBJECT, + properties={ + 'token': openapi.Schema(type=openapi.TYPE_STRING, description='Reset token'), + 'password': openapi.Schema(type=openapi.TYPE_STRING, description='New password'), + }, + required=['token', 'password'], + ), + responses={ + 200: openapi.Response("Success - Password sent successfully."), + 404: openapi.Response("Not Found - The reset password link is expired or invalid.") + } + ) def post(self, request, format='json'): + """ + Handles the POST request to set a new password. + + Validates the provided token and sets the new password if valid. + """ try: serializer = NewPasswordSerializer(data=request.data, context={'request': request}) @@ -2322,10 +3752,18 @@ def post(self, request, format='json'): # Return message. At this point no error have been thrown and this should return success. return Response({}) + @api_view(('POST',)) @renderer_classes((TemplateHTMLRenderer, JSONRenderer)) @csrf_exempt def verify(request): + """ + Verify the OTP token provided by the user. + + This endpoint allows users to verify their one-time password (OTP) using the token sent to their device. + If the OTP is verified successfully, the user’s OTP verification status is updated. + Optionally, users can choose to remember the device for 90 days. + """ try: device = request.user.emaildevice_set.all()[0] data = json.loads(request.body.decode('utf-8')) @@ -2354,6 +3792,11 @@ def verify(request): @renderer_classes((TemplateHTMLRenderer, JSONRenderer)) @csrf_exempt def set_institutional_use(request): + """ + Set the user's institutional use status. + + This endpoint allows users to specify whether their account is being used for institutional purposes. + """ try: data = json.loads(request.body.decode('utf-8')) request.user.institutional_use = data['institutional_use'] @@ -2370,6 +3813,11 @@ def set_institutional_use(request): @renderer_classes((TemplateHTMLRenderer, JSONRenderer)) @csrf_exempt def reset_otp_challenge(request): + """ + Reset the OTP verification challenge for the user. + + This endpoint sends a new OTP challenge to the user's registered device and resets their verification status. + """ from mcserver.utils import send_otp_challenge send_otp_challenge(request.user) @@ -2380,19 +3828,37 @@ def reset_otp_challenge(request): return Response({'otp_challenge_sent': True}) +@csrf_exempt @api_view(('GET',)) @renderer_classes((TemplateHTMLRenderer, JSONRenderer)) -@csrf_exempt def check_otp_verified(request): + """ + Check if the user has verified their OTP. + + This endpoint returns the current OTP verification status of the user. + """ return Response({'otp_verified': request.user.otp_verified}) class UserInstitutionalUseView(APIView): + """ + A view for handling user institutional use information. + """ permission_classes = [IsAuthenticated] serializer_class = UserInstitutionalUseSerializer - + @swagger_auto_schema( + operation_summary="Retrieve User Institutional Use", + responses={ + 200: openapi.Response("Success - Institutional use retrieved successfully."), + 401: openapi.Response("Unauthorized - User must be authenticated."), + 500: openapi.Response("Internal Server Error - Could not retrieve institutional use.") + }, + ) def get(self, request, format='json'): + """ + Retrieve the institutional use data for the authenticated user. + """ try: user = request.user serializer = UserInstitutionalUseSerializer(user) @@ -2403,7 +3869,19 @@ def get(self, request, format='json'): return Response(serializer.data) + @swagger_auto_schema( + operation_summary="Update User Institutional Use", + responses={ + 200: openapi.Response("Success - Institutional use set successfully."), + 400: openapi.Response("Bad Request - Invalid input data."), + 401: openapi.Response("Unauthorized - User must be authenticated."), + 500: openapi.Response("Internal Server Error - Could not set institutional use.") + }, + ) def post(self, request, format='json'): + """ + Update the institutional use data for the authenticated user. + """ try: user = request.user serializer = UserInstitutionalUseSerializer(user, data=request.data) @@ -2418,25 +3896,51 @@ def post(self, request, format='json'): class AnalysisFunctionsListAPIView(ListAPIView): - """ Returns active AnalysisFunction's. """ - permission_classes = (IsAuthenticated, ) + Returns a list of active AnalysisFunctions that are available + to the authenticated user. + """ + permission_classes = (IsAuthenticated,) serializer_class = AnalysisFunctionSerializer def get_queryset(self): user = self.request.user queryset = AnalysisFunction.objects.filter( Q(is_active=True) & ( - Q(only_for_users__isnull=True) | Q(only_for_users=user) + Q(only_for_users__isnull=True) | Q(only_for_users=user) )) return queryset + @swagger_auto_schema( + operation_summary="List active Analysis Functions", + responses={ + 200: openapi.Response("Success - Analysis functions retrieved successfully."), + 403: openapi.Response("Forbidden - Authentication is required.") + } + ) + def list(self, request, *args, **kwargs): + """ + List all active Analysis Functions. + This method overrides the default list method to provide + additional documentation. + """ + return super().list(request, *args, **kwargs) + class InvokeAnalysisFunctionAPIView(APIView): - """ Invokes AnalysisFunction asynchronously with Celery. """ - permission_classes = (IsAuthenticated, ) - + Invokes an Analysis Function asynchronously using Celery. + """ + permission_classes = (IsAuthenticated,) + + @swagger_auto_schema( + operation_summary="Invoke an Analysis Function", + responses={ + 201: openapi.Response("Created - Analysis function created successfully."), + 403: openapi.Response("Forbidden - Authentication is required."), + 404: openapi.Response("Not Found - Analysis fuction not found.") + } + ) def post(self, request, *args, **kwargs): function = get_object_or_404( AnalysisFunction, pk=self.kwargs['pk'], is_active=True @@ -2446,11 +3950,23 @@ def post(self, request, *args, **kwargs): class AnalysisFunctionTaskIdAPIView(APIView): - """ Returns the Celery task id for the analysis function for given trial id. """ - permission_classes = (IsAuthenticated, ) - + Returns the Celery task ID for the analysis function associated with the given trial ID. + """ + permission_classes = (IsAuthenticated,) + + @swagger_auto_schema( + operation_summary="Get Task ID for Analysis Function", + responses={ + 200: openapi.Response("Success - Task ID retrieved successfully."), + 403: openapi.Response("Forbidden - Authentication is required."), + 404: openapi.Response("Not Found - Analysis function or task ID not found.") + } + ) def get(self, request, *args, **kwargs): + """ + Retrieve the task ID associated with the given trial ID for the specified AnalysisFunction. + """ function = get_object_or_404( AnalysisFunction, pk=self.kwargs['pk'], is_active=True ) @@ -2462,18 +3978,30 @@ def get(self, request, *args, **kwargs): class AnalysisResultOnReadyAPIView(APIView): - """ Returns AnalysisResult if it has been proccessed, - otherwise responses with 202 status and makes FE - wait for completion. """ - permission_classes = (IsAuthenticated, ) - + Returns the AnalysisResult if it has been processed; + otherwise, responds with a 202 status, indicating that + the frontend should wait for completion. + """ + permission_classes = (IsAuthenticated,) + + @swagger_auto_schema( + operation_summary="Check Analysis Result Status", + responses={ + 202: openapi.Response("Accepted - The analysis result was accepted and is being processed."), + 404: openapi.Response("Not Found - Analysis result not found."), + 403: openapi.Response("Forbidden - Authentication is required.") + } + ) def get(self, request, *args, **kwargs): + """ + Retrieve the AnalysisResult for a given task ID associated with the authenticated user. + """ result = AnalysisResult.objects.filter( task_id=self.kwargs["task_id"], user=request.user ).first() if result and result.state in ( - AnalysisResultState.SUCCESSFULL, AnalysisResultState.FAILED + AnalysisResultState.SUCCESSFULL, AnalysisResultState.FAILED ): serializer = AnalysisResultSerializer(result) dashboard = AnalysisDashboard.objects.filter( @@ -2491,10 +4019,27 @@ def get(self, request, *args, **kwargs): return Response(data) return Response(status=202) + class AnalysisFunctionsPendingForTrialsAPIView(APIView): - permission_classes = (IsAuthenticated, ) + """ + Returns a list of pending AnalysisResults for trials associated + with the authenticated user. The response includes a mapping + of function IDs to the trial IDs that are pending. + """ + permission_classes = (IsAuthenticated,) + @swagger_auto_schema( + operation_summary="Get Pending Trials for Analysis Functions", + responses={ + 200: openapi.Response("Success - Pending trials for analysis functionsretrieved successfully."), + 403: openapi.Response("Forbidden - Authentication is required.") + } + ) def get(self, request, *args, **kwargs): + """ + Retrieve pending AnalysisResults for trials associated + with the authenticated user. + """ from collections import defaultdict results = AnalysisResult.objects.filter( user=request.user, @@ -2511,9 +4056,25 @@ def get(self, request, *args, **kwargs): class AnalysisFunctionsStatesForTrialsAPIView(APIView): - permission_classes = (IsAuthenticated, ) + """ + Returns the state of AnalysisResults for trials associated + with the authenticated user, including task IDs and dashboard IDs. + Each function ID maps to a dictionary of trial IDs with their states. + """ + permission_classes = (IsAuthenticated,) + @swagger_auto_schema( + operation_summary="Get Analysis Function States for Trials", + responses={ + 200: openapi.Response("Success - Analysis results retrieved successfully."), + 403: openapi.Response("Forbidden - Authentication is required.") + } + ) def get(self, request, *args, **kwargs): + """ + Retrieve the states of AnalysisResults for trials associated + with the authenticated user. + """ from collections import defaultdict results = AnalysisResult.objects.filter(user=request.user).order_by('-id') data = defaultdict(dict) @@ -2531,7 +4092,7 @@ def get(self, request, *args, **kwargs): session_id=result.data['session_id'], name__in=result.data['specific_trial_names']).values_list('id', flat=True) for t_id in trial_ids: - data[result.function_id][str(t_id)] = { + data[result.function_id][str(t_id)] = { 'state': result.state, 'task_id': result.task_id, 'dashboard_id': dashboard_id, @@ -2542,13 +4103,17 @@ def get(self, request, *args, **kwargs): class AnalysisDashboardViewSet(viewsets.ModelViewSet): + """ + Allows authenticated users to retrieve data from the Analysis Dashboard, + including both public and private sessions. It includes actions for retrieving detailed + data related to a specific dashboard. + """ serializer_class = AnalysisDashboardSerializer - permission_classes = [IsPublic | ((IsOwner | IsAdmin | IsBackend))] + permission_classes = [IsPublic | (IsOwner | IsAdmin | IsBackend)] def get_queryset(self): """ - This view should return a list of all the sessions - for the currently authenticated user. + Retrieve the list of sessions for the current user or public sessions. """ user = self.request.user if user.is_authenticated: @@ -2557,6 +4122,32 @@ def get_queryset(self): users_have_public_sessions = User.objects.filter(session__public=True).distinct() return AnalysisDashboard.objects.filter(user__in=users_have_public_sessions) + @swagger_auto_schema( + operation_summary="Get dashboard data", + responses={ + 200: openapi.Response("Success - Analysis dashboard retrieved successfully."), + 403: openapi.Response("Forbidden - Authentication is required."), + 404: openapi.Response("Not Found - Analysis dashboard not found.") + } + ) + @action(detail=True) + def data(self, request, pk): + """ + Retrieve data for a specific dashboard. + """ + dashboard = get_object_or_404(AnalysisDashboard, pk=pk) + if request.user.is_authenticated and request.user == dashboard.user: + return Response(dashboard.get_available_data()) + + return Response(dashboard.get_available_data( + only_public=True, subject_id=request.GET.get('subject_id'), share_token=request.GET.get('share_token'))) + + @swagger_auto_schema( + operation_summary="List dashboards", + responses={ + 200: openapi.Response("Success - List of analysis dashboard retrieved successfully.") + } + ) def list(self, request): queryset = self.get_queryset() if self.request.user.is_authenticated: @@ -2566,11 +4157,73 @@ def list(self, request): serializer = AnalysisDashboardSerializer(queryset, many=True) return Response(serializer.data) - @action(detail=True) - def data(self, request, pk): - dashboard = get_object_or_404(AnalysisDashboard, pk=pk) - if request.user.is_authenticated and request.user == dashboard.user: - return Response(dashboard.get_available_data()) - return Response(dashboard.get_available_data( - only_public=True, subject_id=request.GET.get('subject_id'), share_token=request.GET.get('share_token'))) + @swagger_auto_schema( + operation_summary="Create a dashboard", + request_body=AnalysisDashboardSerializer, + responses={ + 201: openapi.Response('Created - Analysis dashboard created successfully.'), + 400: openapi.Response("Bad Request - Invalid analysis dashboard.") + } + ) + def create(self, request, *args, **kwargs): + """ + Create a new dashboard. + """ + return super().create(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Retrieve a dashboard", + responses={ + 200: openapi.Response("Success - Analysis dashboard retrieved successfully."), + 404: openapi.Response("Not Found - Analysis dashboard not found.") + } + ) + def retrieve(self, request, *args, **kwargs): + """ + Retrieve a specific dashboard. + """ + return super().retrieve(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Update a dashboard", + request_body=AnalysisDashboardSerializer, + responses={ + 200: openapi.Response("Success - Analysis dashboard updated successfully."), + 400: openapi.Response("Bad Request - Invalid analysis dashboard."), + 404: openapi.Response("Not Found - Analysis dashboard not found.") + } + ) + def update(self, request, *args, **kwargs): + """ + Update an existing dashboard. + """ + return super().update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Partially update a dashboard", + request_body=AnalysisDashboardSerializer, + responses={ + 200: openapi.Response("Success - Dashboard partially updated successfully."), + 400: openapi.Response("Bad Request - Invalid analysis dashboard."), + 404: openapi.Response("Not Found - Analysis dashboard not found.") + } + ) + def partial_update(self, request, *args, **kwargs): + """ + Partially update a dashboard. + """ + return super().partial_update(request, *args, **kwargs) + + @swagger_auto_schema( + operation_summary="Delete a dashboard", + responses={ + 204: openapi.Response("Deleted - Analysis dashboard deleted successfully."), + 404: openapi.Response("Not Found - Analysis dashboard not found.") + } + ) + def destroy(self, request, *args, **kwargs): + """ + Delete a dashboard. + """ + return super().destroy(request, *args, **kwargs)