diff --git a/openedx/core/djangoapps/enrollments/views.py b/openedx/core/djangoapps/enrollments/views.py index 2bc565cf7fdf..41aababcdc75 100644 --- a/openedx/core/djangoapps/enrollments/views.py +++ b/openedx/core/djangoapps/enrollments/views.py @@ -1169,3 +1169,67 @@ def list(self, request): response.append(user_progress) return self.get_paginated_response(response) + + +@can_disable_rate_limit +class UserWatchTime(APIView, ApiKeyPermissionMixIn): + authentication_classes = ( + JwtAuthentication, + BearerAuthenticationAllowInactiveUser, + EnrollmentCrossDomainSessionAuth, + ) + permission_classes = (ApiKeyHeaderPermissionIsAuthenticated,) + + def sql_format(template, *args, **kwargs): + args = [sql_escape_string(arg).decode() for arg in args] + kwargs = { + key: sql_escape_string(value).decode() for key, value in kwargs.items() + } + return template.format(*args, **kwargs) + + @method_decorator(ensure_csrf_cookie_cross_domain) + def get(self, request): + """ + Gets a list of all course enrollments for a user. + """ + user_id = request.user.id + clickhouse_uri = ( + f"http://openedx:fIgKZcXnVKWNaLCaJhgN@" + f"cairn-clickhouse:8123/?database=openedx" + ) + query = sql_format( + f"SELECT SUM(duration) `Watch time` FROM `openedx`.`video_view_segments` WHERE user_id={user_id} LIMIT 50000;" + ) + try: + response = requests.get(clickhouse_uri, data=query.encode("utf8")) + watch_time = float(response.content.decode().strip()) / 60 + return Response(status=status.HTTP_200_OK, data={"watch_time": watch_time}) + except Exception as e: + log.error( + f"Unable to fetch watch for user {user_id} due to this exception: {str(e)}" + ) + raise HTTPException(status_code=500, detail=str(e)) + + + +clickhouse_uri = ( + f"http://openedx:fIgKZcXnVKWNaLCaJhgN@" + f"cairn-clickhouse:8123/?database=openedx" +) +def sql_format(template, *args, **kwargs): + args = [sql_escape_string(arg).decode() for arg in args] + kwargs = { + key: sql_escape_string(value).decode() for key, value in kwargs.items() + } + return template.format(*args, **kwargs) + +query = sql_format( + f"SELECT COUNT(*) FROM `openedx`.`video_view_segments`;" +) +response = requests.get(clickhouse_uri, params={'query': query})#data=query.encode("utf8")) +response = response.content.decode() +response + +float(response.strip()) +watch_time = float(response.content.decode().strip()) / 60 +watch_time diff --git a/openedx/features/sdaia_features/course_progress/api/__init__.py b/openedx/features/sdaia_features/course_progress/api/__init__.py new file mode 100644 index 000000000000..e69de29bb2d1 diff --git a/openedx/features/sdaia_features/course_progress/api/v1/__init__.py b/openedx/features/sdaia_features/course_progress/api/v1/__init__.py new file mode 100644 index 000000000000..e69de29bb2d1 diff --git a/openedx/features/sdaia_features/course_progress/api/v1/urls.py b/openedx/features/sdaia_features/course_progress/api/v1/urls.py new file mode 100644 index 000000000000..38ddc36116d2 --- /dev/null +++ b/openedx/features/sdaia_features/course_progress/api/v1/urls.py @@ -0,0 +1,13 @@ +""" +URLs for User Watch Hours - SDAIA Specific. +""" + +from django.urls import path # pylint: disable=unused-import +from .views import UserWatchHoursAPIView + + +app_name = "nafath_api_v1" + +urlpatterns = [ + path(r"user_watch_hours", UserWatchHoursAPIView.as_view()), +] diff --git a/openedx/features/sdaia_features/course_progress/api/v1/views.py b/openedx/features/sdaia_features/course_progress/api/v1/views.py new file mode 100644 index 000000000000..9754d2768a26 --- /dev/null +++ b/openedx/features/sdaia_features/course_progress/api/v1/views.py @@ -0,0 +1,61 @@ +""" +The User Watch Hours API view - SDAIA Specific. +""" + +import logging +import requests + +from django.conf import settings +from django.utils.decorators import method_decorator +from edx_rest_framework_extensions.auth.jwt.authentication import JwtAuthentication +from rest_framework import permissions, status +from rest_framework.response import Response +from rest_framework.views import APIView + +from common.djangoapps.util.disable_rate_limit import can_disable_rate_limit +from openedx.core.djangoapps.cors_csrf.decorators import ensure_csrf_cookie_cross_domain +from openedx.core.lib.api.authentication import BearerAuthenticationAllowInactiveUser +from openedx.core.lib.api.permissions import ApiKeyHeaderPermissionIsAuthenticated + + +log = logging.getLogger(__name__) + + +@can_disable_rate_limit +class UserWatchHoursAPIView(APIView, ApiKeyPermissionMixIn): + authentication_classes = ( + JwtAuthentication, + BearerAuthenticationAllowInactiveUser, + EnrollmentCrossDomainSessionAuth, + ) + permission_classes = (ApiKeyHeaderPermissionIsAuthenticated,) + + def sql_format(template, *args, **kwargs): + args = [sql_escape_string(arg).decode() for arg in args] + kwargs = { + key: sql_escape_string(value).decode() for key, value in kwargs.items() + } + return template.format(*args, **kwargs) + + @method_decorator(ensure_csrf_cookie_cross_domain) + def get(self, request): + """ + Gets the total watch hours for a user. + """ + user_id = request.user.id + clickhouse_uri = ( + f"{settings.CAIRN_CLICKHOUSE_HTTP_SCHEME}://{settings.CAIRN_CLICKHOUSE_USERNAME}:{settings.CAIRN_CLICKHOUSE_PASSWORD}@" + f"{settings.CAIRN_CLICKHOUSE_HOST}:{settings.CAIRN_CLICKHOUSE_HTTP_PORT}/?database={settings.CAIRN_CLICKHOUSE_DATABASE}" + ) + query = self.sql_format( + f"SELECT SUM(duration) as `Watch time` FROM `openedx`.`video_view_segments` WHERE user_id={user_id};" + ) + try: + response = requests.get(clickhouse_uri, data=query.encode("utf8")) + watch_time = float(response.content.decode().strip()) / (60 * 60) + return Response(status=status.HTTP_200_OK, data={"watch_time": watch_time}) + except Exception as e: + log.error( + f"Unable to fetch watch for user {user_id} due to this exception: {str(e)}" + ) + raise HTTPException(status_code=500, detail=str(e)) diff --git a/openedx/features/sdaia_features/course_progress/apps.py b/openedx/features/sdaia_features/course_progress/apps.py index fee8bda6b496..1af0249d67ef 100644 --- a/openedx/features/sdaia_features/course_progress/apps.py +++ b/openedx/features/sdaia_features/course_progress/apps.py @@ -1,19 +1,28 @@ """ Progress Updates App Config """ + from django.apps import AppConfig from edx_django_utils.plugins import PluginURLs, PluginSettings from openedx.core.djangoapps.plugins.constants import ProjectType, SettingsType + class CourseProgressConfig(AppConfig): - name = 'openedx.features.sdaia_features.course_progress' + name = "openedx.features.sdaia_features.course_progress" plugin_app = { + "url_config": { + "lms.djangoapp": { + "namespace": "course_progress", + "regex": r"^sdaia", + "relative_path": "urls", + } + }, PluginSettings.CONFIG: { ProjectType.LMS: { - SettingsType.COMMON: {PluginSettings.RELATIVE_PATH: 'settings.common'}, + SettingsType.COMMON: {PluginSettings.RELATIVE_PATH: "settings.common"}, } - } + }, } def ready(self): diff --git a/openedx/features/sdaia_features/course_progress/urls.py b/openedx/features/sdaia_features/course_progress/urls.py new file mode 100644 index 000000000000..5d9be29a3cdd --- /dev/null +++ b/openedx/features/sdaia_features/course_progress/urls.py @@ -0,0 +1,14 @@ +""" +URLs for User Watch Hours - SDAIA Specific. +""" + +from django.urls import path # pylint: disable=unused-import +from django.conf.urls import include + + +urlpatterns = [ + path( + "/api/v1/", + include("openedx.features.sdaia_features.course_progress.api.v1.urls", namespace="course_progress_api_v1"), + ), +]