Skip to content

Commit

Permalink
Merge pull request #659 from eduNEXT/MJG/restriction-filters
Browse files Browse the repository at this point in the history
feat: backport HomepageRenderStarted and CatalogRenderStarted filters
  • Loading branch information
mariajgrimaldi authored and ErickMurillo committed Feb 12, 2024
1 parent cf6e266 commit 8365f79
Show file tree
Hide file tree
Showing 2 changed files with 192 additions and 11 deletions.
96 changes: 94 additions & 2 deletions common/djangoapps/student/views/management.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,8 @@
from django.db import transaction
from django.db.models.signals import post_save
from django.dispatch import Signal, receiver # lint-amnesty, pylint: disable=unused-import
from django.http import Http404, HttpResponse, HttpResponseBadRequest, HttpResponseForbidden
from django.http import Http404, HttpResponse, HttpResponseBadRequest, HttpResponseForbidden, HttpResponseRedirect

from django.shortcuts import redirect
from django.template.context_processors import csrf
from django.urls import reverse
Expand All @@ -34,6 +35,8 @@
# Note that this lives in LMS, so this dependency should be refactored.
from opaque_keys import InvalidKeyError
from opaque_keys.edx.keys import CourseKey
from openedx_filters.exceptions import OpenEdxFilterException
from openedx_filters.tooling import OpenEdxPublicFilter
from pytz import UTC
from rest_framework.decorators import api_view, authentication_classes, permission_classes
from rest_framework.permissions import IsAuthenticated
Expand Down Expand Up @@ -114,6 +117,79 @@ def csrf_token(context):
' name="csrfmiddlewaretoken" value="{}" /></div>').format(Text(token)))


class HomepageRenderStarted(OpenEdxPublicFilter):
"""
Custom class used to create homepage render filters and its custom methods.
"""

filter_type = "org.openedx.learning.homepage.render.started.v1"

class RedirectToPage(OpenEdxFilterException):
"""
Custom class used to stop the homepage rendering process.
"""

def __init__(self, message, redirect_to=""):
"""
Override init that defines specific arguments used in the homepage render process.
Arguments:
message: error message for the exception.
redirect_to: URL to redirect to.
"""
super().__init__(message, redirect_to=redirect_to)

class RenderInvalidHomepage(OpenEdxFilterException):
"""
Custom class used to stop the homepage render process.
"""

def __init__(self, message, index_template="", template_context=None):
"""
Override init that defines specific arguments used in the index render process.
Arguments:
message: error message for the exception.
index_template: template path rendered instead.
template_context: context used to the new index_template.
"""
super().__init__(
message,
index_template=index_template,
template_context=template_context,
)

class RenderCustomResponse(OpenEdxFilterException):
"""
Custom class used to stop the homepage rendering process.
"""

def __init__(self, message, response=None):
"""
Override init that defines specific arguments used in the homepage render process.
Arguments:
message: error message for the exception.
response: custom response which will be returned by the homepage view.
"""
super().__init__(
message,
response=response,
)

@classmethod
def run_filter(cls, context, template_name):
"""
Execute a filter with the signature specified.
Arguments:
context (dict): context dictionary for homepage template.
template_name (str): template name to be rendered by the homepage.
"""
data = super().run_pipeline(context=context, template_name=template_name)
return data.get("context"), data.get("template_name")


# NOTE: This view is not linked to directly--it is called from
# branding/views.py:index(), which is cached for anonymous users.
# This means that it should always return the same thing for anon
Expand Down Expand Up @@ -168,7 +244,23 @@ def index(request, extra_context=None, user=AnonymousUser()):
# Add marketable programs to the context.
context['programs_list'] = get_programs_with_type(request.site, include_hidden=False)

return render_to_response('index.html', context)
index_template = 'index.html'
try:
# .. filter_implemented_name: HomepageRenderStarted
# .. filter_type: org.openedx.learning.homepage.render.started.v1
context, index_template = HomepageRenderStarted.run_filter(
context=context, template_name=index_template,
)
except HomepageRenderStarted.RenderInvalidHomepage as exc:
response = render_to_response(exc.index_template, exc.template_context) # pylint: disable=no-member
except HomepageRenderStarted.RedirectToPage as exc:
response = HttpResponseRedirect(exc.redirect_to or reverse('account_settings'))
except HomepageRenderStarted.RenderCustomResponse as exc:
response = exc.response # pylint: disable=no-member
else:
response = render_to_response(index_template, context)

return response


def compose_activation_email(user, user_registration=None, route_enabled=False, profile_name='', redirect_url=None):
Expand Down
107 changes: 98 additions & 9 deletions lms/djangoapps/courseware/views/views.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@
from django.core.exceptions import PermissionDenied
from django.db import transaction
from django.db.models import Q, prefetch_related_objects
from django.http import Http404, HttpResponse, HttpResponseBadRequest, HttpResponseForbidden
from django.http import Http404, HttpResponse, HttpResponseBadRequest, HttpResponseForbidden, HttpResponseRedirect
from django.shortcuts import redirect
from django.template.context_processors import csrf
from django.urls import reverse
Expand All @@ -38,6 +38,8 @@
from opaque_keys import InvalidKeyError
from opaque_keys.edx.keys import CourseKey, UsageKey
from openedx_filters.learning.filters import CourseAboutRenderStarted
from openedx_filters.exceptions import OpenEdxFilterException
from openedx_filters.tooling import OpenEdxPublicFilter
from pytz import UTC
from requests.exceptions import ConnectionError, Timeout # pylint: disable=redefined-builtin
from rest_framework import status
Expand Down Expand Up @@ -268,6 +270,79 @@ def user_groups(user):
return group_names


class CatalogRenderStarted(OpenEdxPublicFilter):
"""
Custom class used to create catalog render filters and its custom methods.
"""

filter_type = "org.openedx.learning.catalog.render.started.v1"

class RedirectToPage(OpenEdxFilterException):
"""
Custom class used to stop the catalog rendering process.
"""

def __init__(self, message, redirect_to=""):
"""
Override init that defines specific arguments used in the catalog render process.
Arguments:
message: error message for the exception.
redirect_to: URL to redirect to.
"""
super().__init__(message, redirect_to=redirect_to)

class RenderInvalidCatalog(OpenEdxFilterException):
"""
Custom class used to stop the catalog render process.
"""

def __init__(self, message, courses_template="", template_context=None):
"""
Override init that defines specific arguments used in the index render process.
Arguments:
message: error message for the exception.
courses_template: template path rendered instead.
template_context: context used to the new courses_template.
"""
super().__init__(
message,
courses_template=courses_template,
template_context=template_context,
)

class RenderCustomResponse(OpenEdxFilterException):
"""
Custom class used to stop the catalog rendering process.
"""

def __init__(self, message, response=None):
"""
Override init that defines specific arguments used in the catalog render process.
Arguments:
message: error message for the exception.
response: custom response which will be returned by the catalog view.
"""
super().__init__(
message,
response=response,
)

@classmethod
def run_filter(cls, context, template_name):
"""
Execute a filter with the signature specified.
Arguments:
context (dict): context dictionary for catalog template.
template_name (str): template name to be rendered by the catalog.
"""
data = super().run_pipeline(context=context, template_name=template_name)
return data.get("context"), data.get("template_name")


@ensure_csrf_cookie
@cache_if_anonymous()
def courses(request):
Expand All @@ -288,14 +363,28 @@ def courses(request):
# Add marketable programs to the context.
programs_list = get_programs_with_type(request.site, include_hidden=False)

return render_to_response(
"courseware/courses.html",
{
'courses': courses_list,
'course_discovery_meanings': course_discovery_meanings,
'programs_list': programs_list,
}
)
courses_template = "courseware/courses.html"
context = {
'courses': courses_list,
'course_discovery_meanings': course_discovery_meanings,
'programs_list': programs_list,
}
try:
# .. filter_implemented_name: CatalogRenderStarted
# .. filter_type: org.openedx.learning.catalog.render.started.v1
context, courses_template = CatalogRenderStarted.run_filter(
context=context, template_name=courses_template,
)
except CatalogRenderStarted.RenderInvalidCatalog as exc:
response = render_to_response(exc.courses_template, exc.template_context) # pylint: disable=no-member
except CatalogRenderStarted.RedirectToPage as exc:
response = HttpResponseRedirect(exc.redirect_to or reverse('account_settings'))
except CatalogRenderStarted.RenderCustomResponse as exc:
response = exc.response # pylint: disable=no-member
else:
response = render_to_response(courses_template, context)

return response


class PerUserVideoMetadataThrottle(UserRateThrottle):
Expand Down

0 comments on commit 8365f79

Please sign in to comment.