Files
edx-platform/lms/djangoapps/learner_recommendations/views.py
Syed Sajjad Hussain Shah d694b70439 fix: recommend only available courses (#31773)
VAN-1306

Co-authored-by: Syed Sajjad  Hussain Shah <syed.sajjad@H7FKF7K6XD.local>
2023-02-21 09:04:08 +05:00

154 lines
5.2 KiB
Python

"""
Views for Learner Recommendations.
"""
import logging
from django.conf import settings
from ipware.ip import get_client_ip
from edx_rest_framework_extensions.auth.jwt.authentication import JwtAuthentication
from edx_rest_framework_extensions.auth.session.authentication import (
SessionAuthenticationAllowInactiveUser,
)
from django.core.exceptions import PermissionDenied
from opaque_keys.edx.keys import CourseKey
from rest_framework.permissions import IsAuthenticated
from rest_framework.response import Response
from rest_framework.views import APIView
from common.djangoapps.track import segment
from openedx.core.djangoapps.catalog.utils import (
get_course_data,
get_course_run_details,
)
from openedx.core.djangoapps.geoinfo.api import country_code_from_ip
from openedx.features.enterprise_support.utils import is_enterprise_learner
from lms.djangoapps.learner_recommendations.toggles import enable_course_about_page_recommendations
from lms.djangoapps.learner_recommendations.utils import (
get_algolia_courses_recommendation,
get_amplitude_course_recommendations,
filter_recommended_courses,
)
from lms.djangoapps.learner_recommendations.serializers import RecommendationsSerializer
log = logging.getLogger(__name__)
class AlgoliaCoursesSearchView(APIView):
"""
**Example Request**
GET api/learner_recommendations/algolia/courses/{course_id}/
"""
authentication_classes = (JwtAuthentication, SessionAuthenticationAllowInactiveUser,)
permission_classes = (IsAuthenticated,)
def get(self, request, course_id):
""" Retrieves course recommendations from Algolia based on course skills. """
course_run_data = get_course_run_details(course_id, ["course"])
course_key_str = course_run_data.get("course", None)
# Fetching course level type and skills from discovery service.
course_data = get_course_data(course_key_str, ["level_type", "skill_names"])
# If discovery service fails to fetch data, we will not run recommendations engine.
if not course_data:
return Response({"courses": [], "count": 0}, status=200)
course_data["key"] = course_id
response = get_algolia_courses_recommendation(course_data)
return Response({"courses": response.get("hits", []), "count": response.get("nbHits", 0)}, status=200)
class AmplitudeRecommendationsView(APIView):
"""
**Example Request**
GET api/learner_recommendations/amplitude/{course_id}/
"""
authentication_classes = (JwtAuthentication, SessionAuthenticationAllowInactiveUser,)
permission_classes = (IsAuthenticated,)
recommendations_count = 4
def _emit_recommendations_viewed_event(
self,
user_id,
is_control,
recommended_courses,
amplitude_recommendations=True,
):
"""Emits an event to track recommendation experiment views."""
segment.track(
user_id,
"edx.bi.user.recommendations.viewed",
{
"is_control": is_control,
"amplitude_recommendations": amplitude_recommendations,
"course_key_array": [
course["key"] for course in recommended_courses
],
"page": "course_about_page",
},
)
def get(self, request, course_id):
"""
Returns
- Amplitude course recommendations
- Upsell program if the requesting course has a related program
"""
if not enable_course_about_page_recommendations():
return Response(status=404)
if is_enterprise_learner(request.user):
raise PermissionDenied()
user = request.user
course_locator = CourseKey.from_string(course_id)
course_key = f'{course_locator.org}+{course_locator.course}'
try:
is_control, has_is_control, course_keys = get_amplitude_course_recommendations(
user.id, settings.COURSE_ABOUT_PAGE_AMPLITUDE_RECOMMENDATION_ID
)
except Exception as err: # pylint: disable=broad-except
log.warning(f"Amplitude API failed for {user.id} due to: {err}")
return Response(status=404)
is_control = is_control if has_is_control else None
recommended_courses = []
if not (is_control or is_control is None):
ip_address = get_client_ip(request)[0]
user_country_code = country_code_from_ip(ip_address).upper()
recommended_courses = filter_recommended_courses(
user,
course_keys,
user_country_code=user_country_code,
request_course=course_key,
recommendation_count=self.recommendations_count
)
for course in recommended_courses:
course.update({
"active_course_run": course.get("course_runs")[0]
})
self._emit_recommendations_viewed_event(
user.id, is_control, recommended_courses
)
return Response(
RecommendationsSerializer(
{
"courses": recommended_courses,
"is_control": is_control,
}
).data,
status=200,
)