From 023a9d76bbb3764ecfa5a54e88fd88c941cf87ed Mon Sep 17 00:00:00 2001 From: Adam Stankiewicz Date: Wed, 9 Oct 2024 15:56:58 -0400 Subject: [PATCH] feat: introduce Gateway Aggregation / Backend-for-Frontend abstraction via LearnerPortalBFFAPIView --- enterprise_access/apps/api/v1/urls.py | 5 + .../apps/api/v1/views/__init__.py | 1 + enterprise_access/apps/api/v1/views/bffs.py | 59 +++ .../apps/api_client/base_user.py | 49 ++ .../apps/api_client/license_manager_client.py | 72 +++ enterprise_access/apps/bffs/__init__.py | 0 enterprise_access/apps/bffs/admin.py | 3 + enterprise_access/apps/bffs/apps.py | 6 + enterprise_access/apps/bffs/context.py | 51 ++ enterprise_access/apps/bffs/handlers.py | 452 ++++++++++++++++++ enterprise_access/apps/bffs/models.py | 3 + .../apps/bffs/response_builder.py | 182 +++++++ enterprise_access/apps/bffs/tests.py | 3 + enterprise_access/apps/bffs/views.py | 3 + 14 files changed, 889 insertions(+) create mode 100644 enterprise_access/apps/api/v1/views/bffs.py create mode 100644 enterprise_access/apps/api_client/base_user.py create mode 100644 enterprise_access/apps/bffs/__init__.py create mode 100644 enterprise_access/apps/bffs/admin.py create mode 100644 enterprise_access/apps/bffs/apps.py create mode 100644 enterprise_access/apps/bffs/context.py create mode 100644 enterprise_access/apps/bffs/handlers.py create mode 100644 enterprise_access/apps/bffs/models.py create mode 100644 enterprise_access/apps/bffs/response_builder.py create mode 100644 enterprise_access/apps/bffs/tests.py create mode 100644 enterprise_access/apps/bffs/views.py diff --git a/enterprise_access/apps/api/v1/urls.py b/enterprise_access/apps/api/v1/urls.py index 1357ecf9..2820f773 100644 --- a/enterprise_access/apps/api/v1/urls.py +++ b/enterprise_access/apps/api/v1/urls.py @@ -35,4 +35,9 @@ ), ] +# BFFs +urlpatterns += [ + path('bffs/learner//', views.LearnerPortalBFFAPIView.as_view(), name='learner-portal-bff'), +] + urlpatterns += router.urls diff --git a/enterprise_access/apps/api/v1/views/__init__.py b/enterprise_access/apps/api/v1/views/__init__.py index 7961a1b1..c18f64a3 100644 --- a/enterprise_access/apps/api/v1/views/__init__.py +++ b/enterprise_access/apps/api/v1/views/__init__.py @@ -17,3 +17,4 @@ SubsidyAccessPolicyRedeemViewset, SubsidyAccessPolicyViewSet ) +from .bffs import LearnerPortalBFFAPIView diff --git a/enterprise_access/apps/api/v1/views/bffs.py b/enterprise_access/apps/api/v1/views/bffs.py new file mode 100644 index 00000000..6244b30d --- /dev/null +++ b/enterprise_access/apps/api/v1/views/bffs.py @@ -0,0 +1,59 @@ +""" +Enterprise BFFs for MFEs. +""" + +from rest_framework.response import Response +from rest_framework.views import APIView +from rest_framework.permissions import IsAuthenticated +from rest_framework.authentication import get_authorization_header, SessionAuthentication +from edx_rest_framework_extensions.auth.jwt.authentication import JwtAuthentication +from edx_rest_framework_extensions.auth.jwt.cookies import jwt_cookie_name + +from enterprise_access.apps.bffs.context import HandlerContext +from enterprise_access.apps.bffs.handlers import LearnerPortalHandlerFactory +from enterprise_access.apps.bffs.response_builder import LearnerPortalResponseBuilderFactory + + +class LearnerPortalBFFAPIView(APIView): + """ + API view for learner portal BFF routes. + """ + + authentication_classes = [JwtAuthentication] + permission_classes = [IsAuthenticated] + + def post(self, request, page_route, *args, **kwargs): + """ + Handles GET requests for learner-specific routes. + + Args: + request (Request): The request object. + route (str): The specific learner portal route (e.g., 'dashboard'). + + Returns: + Response: The response data formatted by the response builder. + """ + + # Create the context based on the request + context = HandlerContext(page_route=page_route, request=request) + + # Use the LearnerPortalResponseBuilderFactory to get the appropriate response builder + response_builder = LearnerPortalResponseBuilderFactory.get_response_builder(context) + + try: + # Use the LearnerHandlerFactory to get the appropriate handler + handler = LearnerPortalHandlerFactory.get_handler(context) + + # Load and process data using the handler + handler.load_and_process() + except Exception as exc: + context.add_error( + user_message="An error occurred while processing the request.", + developer_message=f"Error: {exc}", + severity="error", + ) + + # Build the response data and status code + response_data, status_code = response_builder.build() + + return Response(response_data, status=status_code) diff --git a/enterprise_access/apps/api_client/base_user.py b/enterprise_access/apps/api_client/base_user.py new file mode 100644 index 00000000..6f0fec82 --- /dev/null +++ b/enterprise_access/apps/api_client/base_user.py @@ -0,0 +1,49 @@ +import crum +import requests + +from edx_django_utils.monitoring import set_custom_attribute +from edx_rest_framework_extensions.auth.jwt.cookies import jwt_cookie_name + + +def get_request_id(): + """ + Helper to get the request id - usually set via an X-Request-ID header + """ + request = crum.get_current_request() + if request is not None and request.headers is not None: + return request.headers.get('X-Request-ID') + else: + return None + + +class BaseUserApiClient(requests.Session): + """ + A requests Session that includes the Authorization and User-Agent headers from the original request. + """ + def __init__(self, original_request, **kwargs): + super().__init__(**kwargs) + self.original_request = original_request + + self.headers = {} + + if self.original_request: + # If no Authorization header, check for JWT in cookies + jwt_token = self.original_request.COOKIES.get(jwt_cookie_name()) + if 'Authorization' not in self.headers and jwt_token is not None: + self.headers['Authorization'] = f'JWT {jwt_token}' + + # Add X-Request-ID header if applicable + request_id = get_request_id() + if self.headers.get('X-Request-ID') is None and request_id is not None: + self.headers['X-Request-ID'] = request_id + + def request(self, method, url, headers=None, **kwargs): # pylint: disable=arguments-differ + if headers: + headers.update(self.headers) + else: + headers = self.headers + + # Set `api_client` as a custom attribute for monitoring, reflecting the API client's module path + set_custom_attribute('api_client', 'enterprise_access.apps.api_client.base_user.BaseUserApiClient') + + return super().request(method, url, headers=headers, **kwargs) diff --git a/enterprise_access/apps/api_client/license_manager_client.py b/enterprise_access/apps/api_client/license_manager_client.py index 383bc54a..be86bee8 100644 --- a/enterprise_access/apps/api_client/license_manager_client.py +++ b/enterprise_access/apps/api_client/license_manager_client.py @@ -7,6 +7,7 @@ from django.conf import settings from enterprise_access.apps.api_client.base_oauth import BaseOAuthClient +from enterprise_access.apps.api_client.base_user import BaseUserApiClient logger = logging.getLogger(__name__) @@ -62,3 +63,74 @@ def assign_licenses(self, user_emails, subscription_uuid): except requests.exceptions.HTTPError as exc: logger.exception(exc) raise + + +class LicenseManagerUserApiClient(BaseUserApiClient): + """ + API client for calls to the license-manager service. This client is used for user-specific calls, + passing the original Authorization header from the originating request. + """ + + api_base_url = f"{settings.LICENSE_MANAGER_URL}/api/v1/" + learner_licenses_endpoint = f"{api_base_url}learner-licenses/" + license_activation_endpoint = f"{api_base_url}license-activation/" + + def auto_apply_license_endpoint(self, customer_agreement_uuid): + return f"{self.api_base_url}customer-agreement/{customer_agreement_uuid}/auto-apply/" + + def get_subscription_licenses_for_learner(self, enterprise_customer_uuid): + """ + Get subscription licenses for a learner. + + Arguments: + enterprise_customer_uuid (str): UUID of the enterprise customer + Returns: + dict: Dictionary representation of json returned from API + """ + query_params = { + 'enterprise_customer_uuid': enterprise_customer_uuid, + } + url = self.learner_licenses_endpoint + try: + response = self.get(url, params=query_params, timeout=settings.LICENSE_MANAGER_CLIENT_TIMEOUT) + return response.json() + except requests.exceptions.HTTPError as exc: + logger.exception(f"Failed to get subscription licenses for learner: {exc}") + raise + + def activate_license(self, activation_key): + """ + Activate a license. + + Arguments: + license_uuid (str): UUID of the license to activate + """ + try: + url = self.license_activation_endpoint + query_params = { + 'activation_key': activation_key, + } + response = self.post(url, params=query_params, timeout=settings.LICENSE_MANAGER_CLIENT_TIMEOUT) + response.raise_for_status() + if response.status_code == 204: # Response contains no content + return None + return response.json() + except requests.exceptions.HTTPError as exc: + logger.exception(f"Failed to activate license: {exc}") + raise + + def auto_apply_license(self, customer_agreement_uuid): + """ + Activate a license. + + Arguments: + license_uuid (str): UUID of the license to activate + """ + try: + url = self.auto_apply_license_endpoint(customer_agreement_uuid=customer_agreement_uuid) + response = self.post(url, timeout=settings.LICENSE_MANAGER_CLIENT_TIMEOUT) + response.raise_for_status() + return response.json() + except requests.exceptions.HTTPError as exc: + logger.exception(f"Failed to auto-apply license: {exc}") + raise diff --git a/enterprise_access/apps/bffs/__init__.py b/enterprise_access/apps/bffs/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/enterprise_access/apps/bffs/admin.py b/enterprise_access/apps/bffs/admin.py new file mode 100644 index 00000000..8c38f3f3 --- /dev/null +++ b/enterprise_access/apps/bffs/admin.py @@ -0,0 +1,3 @@ +from django.contrib import admin + +# Register your models here. diff --git a/enterprise_access/apps/bffs/apps.py b/enterprise_access/apps/bffs/apps.py new file mode 100644 index 00000000..a71a84e8 --- /dev/null +++ b/enterprise_access/apps/bffs/apps.py @@ -0,0 +1,6 @@ +from django.apps import AppConfig + + +class BffsConfig(AppConfig): + default_auto_field = 'django.db.models.BigAutoField' + name = 'enterprise_access.apps.bffs' diff --git a/enterprise_access/apps/bffs/context.py b/enterprise_access/apps/bffs/context.py new file mode 100644 index 00000000..71206b24 --- /dev/null +++ b/enterprise_access/apps/bffs/context.py @@ -0,0 +1,51 @@ +""" +HandlerContext for bffs app. +""" + +class HandlerContext: + """ + A context object for managing the state throughout the lifecycle of a Backend-for-Frontend (BFF) request. + + The `HandlerContext` class stores request information, the current route, loaded data, and any errors + that may occur during the request. + + Attributes: + request: The original request object containing information about the incoming HTTP request. + route: The route for which the response is being generated. + data: A dictionary to store data loaded and processed by the handlers. + errors: A list to store errors that occur during request processing. + """ + + def __init__(self, request, page_route): + """ + Initializes the HandlerContext with request information, route, and optional initial data. + + Args: + request: The incoming HTTP request. + page_route: The route identifier for the request. + """ + self.page_route = page_route + self.request = request + self.user = request.user + self.data = {} # Stores processed data for the response + self.errors = [] # Stores any errors that occur during processing + self.enterprise_customer_uuid = None + self.lms_user_id = None + + def add_error(self, user_message, developer_message, severity='error'): + """ + Adds an error to the context. + + Args: + user_message (str): A user-friendly error message. + developer_message (str): A more detailed error message for debugging purposes. + severity (str): The severity level of the error ('error' or 'warning'). Defaults to 'error'. + """ + if not (user_message and developer_message): + raise ValueError("User message and developer message are required for errors.") + + self.errors.append({ + "user_message": user_message, + "developer_message": developer_message, + "severity": severity, + }) diff --git a/enterprise_access/apps/bffs/handlers.py b/enterprise_access/apps/bffs/handlers.py new file mode 100644 index 00000000..a7dd5fc2 --- /dev/null +++ b/enterprise_access/apps/bffs/handlers.py @@ -0,0 +1,452 @@ +"""" +Handlers for bffs app. +""" + +import logging + +# from enterprise_access.apps.api_client.lms_client import LmsApiClient +from enterprise_access.apps.api_client.license_manager_client import LicenseManagerUserApiClient +from enterprise_access.apps.bffs.context import HandlerContext +from enterprise_access.utils import localized_utcnow + +logger = logging.getLogger(__name__) + + +class BaseHandler: + """ + A base handler class that provides shared core functionality for different BFF handlers. + + The `BaseHandler` includes core methods for loading data and adding errors to the context. + Specific handlers, like `LearnerPortalRouteHandler` should extend this class. + """ + + def __init__(self, context: HandlerContext, params=None): + """ + Initializes the BaseHandler with a HandlerContext. + + Args: + context (HandlerContext): The context object containing request information and data. + params (dict): Additional request parameters. Defaults to None. + """ + self.context = context + self.params = params if params else {} + + # Initialize API clients + self.license_manager_client = LicenseManagerUserApiClient(context.request) + + # Set common context attributes + self.initialize_common_context_data() + + def load_and_process(self): + """ + Loads and processes data. This method should be overridden by subclasses to implement + specific data loading and transformation logic. + """ + raise NotImplementedError("Subclasses must implement `load_and_process` method.") + + def add_error(self, user_message, developer_message, severity='error'): + """ + Adds an error to the context. + + Args: + user_message (str): A user-friendly error message. + developer_message (str): A more detailed error message for debugging purposes. + severity (str): The severity level of the error ('error' or 'warning'). Defaults to 'error'. + """ + self.context.add_error(user_message, developer_message, severity) + + def initialize_common_context_data(self): + """ + Initialize commonly used context attributes, such as enterprise customer UUID and LMS user ID. + """ + # Set enterprise_customer_uuid from request parameters or previously set context + enterprise_customer_uuid = ( + self.params.get('enterprise_customer_uuid') \ + or self.context.request.query_params.get('enterprise_customer_uuid') \ + or self.context.request.data.get('enterprise_customer_uuid') + ) + if enterprise_customer_uuid: + self.context.enterprise_customer_uuid = enterprise_customer_uuid + else: + raise ValueError("enterprise_customer_uuid is required for this request.") + + # Set lms_user_id from the authenticated user object in the request + if hasattr(self.context.user, 'lms_user_id)'): + self.context.lms_user_id = self.context.user.lms_user_id + + +class BaseLearnerPortalHandler(BaseHandler): + """ + A base handler class for learner-focused routes. + + The `BaseLearnerHandler` extends `BaseHandler` and provides shared core functionality + across all learner-focused page routes, such as the learner dashboard, search, and course routes. + """ + + def load_and_process(self): + """ + Loads and processes data. This is a basic implementation that can be overridden by subclasses. + + The method in this class simply calls common learner logic to ensure the context is set up. + """ + try: + # Retrieve and process subscription licenses. Handles activation and auto-apply logic. + self.load_subscription_licenses() + self.process_subscription_licenses() + + # Retrieve default enterprise courses and enroll in the redeemable ones + self.load_default_enterprise_courses() + self.enroll_in_redeemable_default_courses() + except Exception as e: + self.add_error( + user_message="An error occurred while loading and processing common learner logic.", + developer_message=f"Error: {str(e)}", + severity='error' + ) + + def load_subscription_licenses(self): + """ + Load subscription licenses for the learner. + """ + subscriptions_result = self.license_manager_client.get_subscription_licenses_for_learner( + enterprise_customer_uuid=self.context.enterprise_customer_uuid + ) + self.transform_subscriptions_result(subscriptions_result) + + def get_subscription_licenses(self): + """ + Get subscription licenses. + """ + return self.context.data['subscriptions'].get('subscription_licenses', []) + + def get_subscription_licenses_by_status(self): + """ + Get subscription licenses by status. + """ + return self.context.data['subscriptions'].get('subscription_licenses_by_status', {}) + + def transform_subscription_licenses(self, subscription_licenses): + """ + Transform subscription licenses data if needed. + """ + return [ + { + 'uuid': subscription_license.get('uuid'), + 'status': subscription_license.get('status'), + 'user_email': subscription_license.get('user_email'), + 'activation_date': subscription_license.get('activation_date'), + 'last_remind_date': subscription_license.get('last_remind_date'), + 'revoked_date': subscription_license.get('revoked_date'), + 'activation_key': subscription_license.get('activation_key'), + 'subscription_plan': subscription_license.get('subscription_plan', {}), + } + for subscription_license in subscription_licenses + ] + + def transform_subscriptions_result(self, subscriptions_result): + """ + Transform subscription licenses data if needed. + """ + subscription_licenses = subscriptions_result.get('results', []) + subscription_licenses_by_status = {} + + transformed_licenses = self.transform_subscription_licenses(subscription_licenses) + + for subscription_license in transformed_licenses: + status = subscription_license.get('status') + if status not in subscription_licenses_by_status: + subscription_licenses_by_status[status] = [] + + subscription_licenses_by_status[status].append({ + 'uuid': subscription_license.get('uuid'), + 'status': status, + 'user_email': subscription_license.get('user_email'), + 'activation_date': subscription_license.get('activation_date'), + 'last_remind_date': subscription_license.get('last_remind_date'), + 'revoked_date': subscription_license.get('revoked_date'), + 'activation_key': subscription_license.get('activation_key'), + 'subscription_plan': subscription_license.get('subscription_plan', {}), + }) + + subscriptions_data = { + 'customer_agreement': subscriptions_result.get('customer_agreement', {}), + 'subscription_licenses': transformed_licenses, + 'subscription_licenses_by_status': subscription_licenses_by_status, + } + self.context.data['subscriptions'] = subscriptions_data + + def check_has_activated_license(self): + """ + Check if the user has an activated license. + + Args: + subscription_licenses_by_status (dict): A dictionary of subscription licenses by status. + + Returns: + bool: True if the user has an activated license, False otherwise. + """ + subscription_licenses_by_status = self.get_subscription_licenses_by_status() + return bool(subscription_licenses_by_status.get('activated')) + + def process_subscription_licenses(self): + """ + Process loaded subscription licenses, including performing side effects such as activation. + + This method is called after `load_subscription_licenses` to handle further actions based + on the loaded data. + """ + # Check if user already has 'activated' license(s). If so, no further action is needed. + if self.check_has_activated_license(): + return + + # Check if there are 'assigned' licenses that need to be activated + self.check_and_activate_assigned_license() + + # Check if there user should be auto-applied a license + self.check_and_auto_apply_license() + + def check_and_activate_assigned_license(self): + """ + Check if there are assigned licenses that need to be activated. + """ + subscription_licenses = self.get_subscription_licenses() + subscription_licenses_by_status = self.get_subscription_licenses_by_status() + assigned_licenses = subscription_licenses_by_status.get('assigned', []) + activated_licenses = [] + for subscription_license in assigned_licenses: + activation_key = subscription_license.get('activation_key') + if activation_key: + try: + # Perform side effect: Activate the assigned license + self.license_manager_client.activate_license(activation_key) + except Exception as e: + logger.exception(f"Error activating license {subscription_license.get('uuid')}: {str(e)}") + self.add_error( + user_message="An error occurred while activating a subscription license.", + developer_message=f"License UUID: {subscription_license.get('uuid')}, Error: {str(e)}", + severity='error' + ) + return + + # Update the subscription_license data with the activation status and date; the activated license is not + # returned from the API, so we need to manually update the license object we have available. + subscription_license['status'] = 'activated' + subscription_license['activation_date'] = localized_utcnow() + activated_licenses.append(subscription_license) + else: + logger.error(f"Activation key not found for license {subscription_license.get('uuid')}") + self.add_error( + user_message="An error occurred while activating a subscription license.", + developer_message=f"Activation key not found for license {subscription_license.get('uuid')}", + severity='error' + ) + + # Update the subscriptions.subscription_licenses_by_status context with the modified licenses data + updated_activated_licenses = subscription_licenses_by_status.get('activated', []) + updated_activated_licenses.extend(activated_licenses) + subscription_licenses_by_status['activated'] = updated_activated_licenses + remaining_assigned_licenses = [ + subscription_license + for subscription_license in assigned_licenses + if subscription_license not in activated_licenses + ] + if remaining_assigned_licenses: + subscription_licenses_by_status['assigned'] = remaining_assigned_licenses + else: + subscription_licenses_by_status.pop('assigned', None) + self.context.data['subscriptions']['subscription_licenses_by_status'] = subscription_licenses_by_status + + # Update the subscriptions.subscription_licenses context with the modified licenses data + updated_subscription_licenses = [] + for subscription_license in subscription_licenses: + for activated_license in activated_licenses: + if subscription_license.get('uuid') == activated_license.get('uuid'): + updated_subscription_licenses.append(activated_license) + break + else: + updated_subscription_licenses.append(subscription_license) + self.context.data['subscriptions']['subscription_licenses'] = updated_subscription_licenses + + def check_and_auto_apply_license(self): + """ + Check if auto-apply licenses are available and apply them to the user. + + Args: + subscription_licenses_by_status (dict): A dictionary of subscription licenses by status. + """ + subscription_licenses_by_status = self.get_subscription_licenses_by_status() + has_assigned_licenses = subscription_licenses_by_status.get('assigned', []) + if has_assigned_licenses or self.check_has_activated_license(): + # Skip auto-apply if user already has an activated license or assigned licenses + return + + customer_agreement = self.context.data['subscriptions'].get('customer_agreement', {}) + has_subscription_plan_for_auto_apply = ( + bool(customer_agreement.get('subscription_for_auto_applied_licenses')) + and customer_agreement.get('net_days_until_expiration') > 0 + ) + idp_or_univeral_link_enabled = ( + # TODO: IDP from customer + customer_agreement.get('enable_auto_applied_subscriptions_with_universal_link') + ) + is_eligible_for_auto_apply = has_subscription_plan_for_auto_apply and idp_or_univeral_link_enabled + if not is_eligible_for_auto_apply: + # Skip auto-apply if the customer agreement does not have a subscription plan for auto-apply + return + + try: + # Perform side effect: Auto-apply license + auto_applied_license = self.license_manager_client.auto_apply_license(customer_agreement.get('uuid')) + if auto_applied_license: + # Update the context with the auto-applied license data + subscription_licenses_by_status['activated'] =\ + self.transform_subscription_licenses([auto_applied_license]) + self.context.data['subscriptions']['subscription_licenses_by_status'] = subscription_licenses_by_status + except Exception as e: + logger.exception(f"Error auto-applying license: {str(e)}") + self.add_error( + user_message="An error occurred while auto-applying a license.", + developer_message=f"Customer agreement UUID: {customer_agreement.get('uuid')}, Error: {str(e)}", + severity='error' + ) + + def load_default_enterprise_courses(self): + """ + Load default enterprise course enrollments (stubbed) + """ + mock_catalog_uuid = 'f09ff39b-f456-4a03-b53b-44cd70f52108' + + self.context.data['default_enterprise_courses'] = [ + { + 'current_course_run_key': 'course-v1:edX+DemoX+Demo_Course', + 'applicable_catalog_uuids': [mock_catalog_uuid], + }, + { + 'current_course_run_key': 'course-v1:edX+SampleX+Sample_Course', + 'applicable_catalog_uuids': [mock_catalog_uuid], + }, + ] + + def enroll_in_redeemable_default_courses(self): + """ + Enroll in redeemable courses. + """ + default_enterprise_courses = self.context.data.get('default_enterprise_courses', []) + activated_subscription_licenses = self.get_subscription_licenses_by_status().get('activated', []) + + if not (default_enterprise_courses or activated_subscription_licenses): + # Skip enrollment if there are no default enterprise courses or activated subscription licenses + return + + redeemable_default_courses = [] + for course in default_enterprise_courses: + for subscription_license in activated_subscription_licenses: + subscription_plan = subscription_license.get('subscription_plan', {}) + if subscription_plan.get('enterprise_catalog_uuid') in course.get('applicable_catalog_uuids'): + redeemable_default_courses.append((course, subscription_license)) + break + + for redeemable_course, subscription_license in redeemable_default_courses: + # Enroll in redeemable courses (stubbed) + if not self.context.data.get('enrolled_default_courses'): + self.context.data['enrolled_default_courses'] = [] + + self.context.data['enrolled_default_courses'].append({ + 'course_key': redeemable_course.get('key'), + 'enrollment_status': 'enrolled', + 'subscription_license_uuid': subscription_license.get('uuid'), + }) + + +class DashboardHandler(BaseLearnerPortalHandler): + """ + A handler class for processing the learner dashboard route. + + The `DashboardHandler` extends `BaseLearnerPortalHandler` to handle the loading and processing + of data specific to the learner dashboard. + """ + + def load_and_process(self): + """ + Loads and processes data for the learner dashboard route. + + This method overrides the `load_and_process` method in `BaseLearnerPortalHandler`. + """ + # Call the common learner logic from the base class + super().load_and_process() + + try: + # Load data specific to the dashboard route + self.context.data['enterprise_course_enrollments'] = self.get_enterprise_course_enrollments() + except Exception as e: + self.add_error( + user_message="An error occurred while processing the learner dashboard.", + developer_message=f"Error: {str(e)}", + severity='error' + ) + + def get_enterprise_course_enrollments(self): + """ + Loads enterprise course enrollments data. + + Returns: + list: A list of enterprise course enrollments. + """ + # Placeholder logic for loading enterprise course enrollments data + return [ + { + "certificate_download_url": None, + "emails_enabled": False, + "course_run_id": "course-v1:BabsonX+MIS01x+1T2019", + "course_run_status": "in_progress", + "created": "2023-09-29T14:24:45.409031+00:00", + "start_date": "2019-03-19T10:00:00Z", + "end_date": "2024-12-31T04:30:00Z", + "display_name": "AI for Leaders", + "course_run_url": "https://learning.edx.org/course/course-v1:BabsonX+MIS01x+1T2019/home", + "due_dates": [], + "pacing": "self", + "org_name": "BabsonX", + "is_revoked": False, + "is_enrollment_active": True, + "mode": "verified", + "resume_course_run_url": None, + "course_key": "BabsonX+MIS01x", + "course_type": "verified-audit", + "product_source": "edx", + "enroll_by": "2024-12-21T23:59:59Z" + } + ] + + +class LearnerPortalHandlerFactory: + """ + Factory to create learner handlers based on route information. + + The `LearnerPortalHandlerFactory` provides a method to instantiate appropriate learner handlers + based on the route stored in the HandlerContext. + """ + + @staticmethod + def get_handler(context): + """ + Returns a route-specific learner handler based on the route information in the context. + + Args: + context (HandlerContext): The context object containing data, errors, and route information. + + Returns: + BaseLearnerHandler: An instance of the appropriate learner handler class. + + Raises: + ValueError: If no learner handler is found for the given route. + """ + page_route = context.page_route + + if page_route == 'dashboard': + return DashboardHandler(context) + elif page_route == 'course': + # Placeholder for CourseHandler, to be implemented similarly to DashboardHandler + raise NotImplementedError("CourseHandler not yet implemented.") + else: + raise ValueError(f"No learner portal handler found for page route: {page_route}") diff --git a/enterprise_access/apps/bffs/models.py b/enterprise_access/apps/bffs/models.py new file mode 100644 index 00000000..71a83623 --- /dev/null +++ b/enterprise_access/apps/bffs/models.py @@ -0,0 +1,3 @@ +from django.db import models + +# Create your models here. diff --git a/enterprise_access/apps/bffs/response_builder.py b/enterprise_access/apps/bffs/response_builder.py new file mode 100644 index 00000000..3818299d --- /dev/null +++ b/enterprise_access/apps/bffs/response_builder.py @@ -0,0 +1,182 @@ +""" +TODO +""" + + +class BaseResponseBuilder: + """ + A base response builder class that provides shared core functionality for different response builders. + + The `BaseResponseBuilder` includes methods for building response data and can be extended by specific + response builders like `LearnerDashboardResponseBuilder` or `CourseResponseBuilder`. + """ + + def __init__(self, context): + """ + Initializes the BaseResponseBuilder with a HandlerContext. + + Args: + context (HandlerContext): The context object containing data, errors, and request information. + """ + self.context = context + + def build(self): + """ + Builds the response data. This method should be overridden by subclasses to implement + specific response formatting logic. + + Returns: + dict: A dictionary containing the response data. + """ + raise NotImplementedError("Subclasses must implement the `build` method.") + + def add_errors_to_response(self, response_data): + """ + Adds any errors to the response data. + """ + if self.context.errors: + response_data['errors'] = [ + error for error in self.context.errors if error['severity'] == 'error' + ] + response_data['warnings'] = [ + error for error in self.context.errors if error['severity'] == 'warning' + ] + return response_data + + def get_status_code(self): + """ + Gets the current status code from the context. + + Returns: + int: The HTTP status code. + """ + return self.context.status_code if hasattr(self.context, 'status_code') else 200 + + +class BaseLearnerResponseBuilder(BaseResponseBuilder): + """ + A base response builder class for learner-focused routes. + + The `BaseLearnerResponseBuilder` extends `BaseResponseBuilder` and provides shared core functionality + for building responses across all learner-focused page routes. + """ + + def common_response_logic(self, response_data): + """ + Applies common response logic for learner-related responses. + + Args: + response_data (dict): The initial response data. + + Returns: + dict: The modified response data with common logic applied. + """ + subscriptions_context = self.context.data.get('subscriptions', {}) + enterprise_customer_user_subsidies = response_data.get('enterprise_customer_user_subsidies', {}) + subscriptions = enterprise_customer_user_subsidies.get('subscriptions', {}) + subscriptions.update(subscriptions_context) + enterprise_customer_user_subsidies.update({ + 'subscriptions': subscriptions, + }) + response_data['enterprise_customer_user_subsidies'] = enterprise_customer_user_subsidies + return response_data + + def build(self): + """ + Builds the base response data for learner routes. + + This method can be overridden by subclasses to provide route-specific logic. + + Returns: + dict: A dictionary containing the base response data. + """ + # Initialize response data with common learner-related logic + response_data = {} + response_data = self.common_response_logic(response_data) + + # Add any errors, etc. + response_data = self.add_errors_to_response(response_data) + + return response_data + + +class LearnerDashboardResponseBuilder(BaseLearnerResponseBuilder): + """ + A response builder for the learner dashboard route. + + The `LearnerDashboardResponseBuilder` extends `BaseLearnerResponseBuilder` to extract and format data + relevant to the learner dashboard page. + """ + + def build(self): + """ + Builds the response data for the learner dashboard route. + + This method overrides the `build` method in `BaseResponseBuilder`. + + Returns: + dict: A dictionary containing the learner dashboard response data. + """ + # Initialize the response data with common learner-related fields + response_data = self.common_response_logic({}) + + # Add specific fields related to the learner dashboard + response_data.update({ + 'enterprise_course_enrollments': self.context.data.get('enterprise_course_enrollments', {}), + }) + + # Add any errors and warnings to the response + response_data = self.add_errors_to_response(response_data) + + # Retrieve the status code + status_code = self.get_status_code() + + return response_data, status_code + + +class BaseResponseBuilderFactory: + """ + A base factory to create response builders based on route information. + + The `BaseResponseBuilderFactory` provides a method to instantiate appropriate response + builders based on route information, allowing for shared logic between specialized factories. + """ + + _response_builder_map = {} + + @classmethod + def get_response_builder(cls, context): + """ + Returns a route-specific response builder based on the route information in the context. + + Args: + context (HandlerContext): The context object containing data, errors, and route information. + + Returns: + BaseResponseBuilder: An instance of the appropriate response builder class. + + Raises: + ValueError: If no response builder is found for the given route. + """ + page_route = context.page_route + + response_builder_class = cls._response_builder_map.get(page_route) + + if response_builder_class is not None: + return response_builder_class(context) + + raise ValueError(f"No response builder found for route: {page_route}") + + +class LearnerPortalResponseBuilderFactory(BaseResponseBuilderFactory): + """ + A learner portal-specific factory to create response builders based on learner portal route information. + + The `LearnerPortalResponseBuilderFactory` extends `BaseResponseBuilderFactory` and provides a + mapping of learner portal-specific routes to response builders. + """ + + _response_builder_map = { + 'dashboard': LearnerDashboardResponseBuilder, + # Add additional routes and response builders here + } diff --git a/enterprise_access/apps/bffs/tests.py b/enterprise_access/apps/bffs/tests.py new file mode 100644 index 00000000..7ce503c2 --- /dev/null +++ b/enterprise_access/apps/bffs/tests.py @@ -0,0 +1,3 @@ +from django.test import TestCase + +# Create your tests here. diff --git a/enterprise_access/apps/bffs/views.py b/enterprise_access/apps/bffs/views.py new file mode 100644 index 00000000..91ea44a2 --- /dev/null +++ b/enterprise_access/apps/bffs/views.py @@ -0,0 +1,3 @@ +from django.shortcuts import render + +# Create your views here.