diff --git a/credentials/apps/core/admin.py b/credentials/apps/core/admin.py index 2f83551aa..cc51eb681 100644 --- a/credentials/apps/core/admin.py +++ b/credentials/apps/core/admin.py @@ -27,7 +27,7 @@ class SiteConfigurationAdmin(admin.ModelAdmin): search_fields = ('site__name',) form = SiteConfigurationAdminForm fieldsets = ( - (None, {'fields': ('site', 'platform_name', 'company_name', 'segment_key', 'theme_name', + (None, {'fields': ('site', 'edx_org_short_name', 'platform_name', 'company_name', 'segment_key', 'theme_name', 'partner_from_address', 'records_enabled',)}), (_('URLs'), { 'fields': ( diff --git a/credentials/apps/core/migrations/0021_siteconfiguration_edx_org_short_name.py b/credentials/apps/core/migrations/0021_siteconfiguration_edx_org_short_name.py new file mode 100644 index 000000000..4dd9af949 --- /dev/null +++ b/credentials/apps/core/migrations/0021_siteconfiguration_edx_org_short_name.py @@ -0,0 +1,20 @@ +# -*- coding: utf-8 -*- +# Generated by Django 1.11.29 on 2020-11-30 07:23 +from __future__ import unicode_literals + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('core', '0020_siteconfiguration_edly_client_branding_and_django_settings'), + ] + + operations = [ + migrations.AddField( + model_name='siteconfiguration', + name='edx_org_short_name', + field=models.CharField(help_text='Unique, short string identifier for organization, same as LMS. Please do not use spaces or special characters. Only allowed special characters are period (.), hyphen (-) and underscore (_).', max_length=255, null=True, unique=True, verbose_name='Organization short name'), + ), + ] diff --git a/credentials/apps/core/models.py b/credentials/apps/core/models.py index 0ccd42fe4..368a6e893 100644 --- a/credentials/apps/core/models.py +++ b/credentials/apps/core/models.py @@ -19,6 +19,17 @@ class SiteConfiguration(models.Model): site = models.OneToOneField(Site, null=False, blank=False) + edx_org_short_name = models.CharField( + max_length=255, + unique=True, + verbose_name=u'Organization short name', + help_text=_( + 'Unique, short string identifier for organization, same as LMS. ' + 'Please do not use spaces or special characters. ' + 'Only allowed special characters are period (.), hyphen (-) and underscore (_).' + ), + null=True, + ) platform_name = models.CharField( verbose_name=_('Platform Name'), help_text=_('Name of your Open edX platform'), diff --git a/credentials/apps/core/tests/factories.py b/credentials/apps/core/tests/factories.py index 4f8a3d330..611a76bff 100644 --- a/credentials/apps/core/tests/factories.py +++ b/credentials/apps/core/tests/factories.py @@ -37,6 +37,7 @@ class Meta(object): model = SiteConfiguration site = SubFactory(SiteFactory) + edx_org_short_name = Faker('word') lms_url_root = Faker('url') catalog_api_url = Faker('url') platform_name = Faker('word') diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/__init__.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/fields.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/fields.py new file mode 100644 index 000000000..15cfcda31 --- /dev/null +++ b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/fields.py @@ -0,0 +1,26 @@ +import base64 + +from django.core.files.base import ContentFile +from rest_framework import serializers + + +class Base64ImageField(serializers.ImageField): + """ + Custom serializer field to ingest base64 encoded image. + """ + + def to_internal_value(self, data): + """ + Save base 64 encoded images. + + SOURCE: http://matthewdaly.co.uk/blog/2015/07/04/handling-images-as-base64-strings-with-django-rest-framework/ + """ + if not data: + return None + + if isinstance(data, str) and data.startswith('data:image'): + file_format, image_string = data.split(';base64,') # format ~=  + file_extension = file_format.split('/')[-1] + data = ContentFile(base64.b64decode(image_string), name='tmp.' + file_extension) + + return super(Base64ImageField, self).to_internal_value(data) diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/permissions.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/permissions.py new file mode 100644 index 000000000..b82bf746d --- /dev/null +++ b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/permissions.py @@ -0,0 +1,25 @@ +""" +Edly's Custom permissions classes for use with DRF. +""" +import logging + +from rest_framework import permissions + +from credentials.apps.edx_credentials_extensions.edly_credentials_app.utils import user_has_edx_organization_access + +logger = logging.getLogger(__name__) + + +class CanAccessCurrentSite(permissions.BasePermission): + """ + Permission to check if the current site is allowed for the user. + """ + + def has_permission(self, request, view): + """ + Checks for user's permission for current site. + """ + if user_has_edx_organization_access(request): + return True + + return False diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/serializers.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/serializers.py new file mode 100644 index 000000000..3090bcd0d --- /dev/null +++ b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/serializers.py @@ -0,0 +1,63 @@ +""" +Serializers for edly_api. +""" +from django.contrib.sites.models import Site + +from rest_framework import serializers + +from credentials.apps.credentials.models import ProgramCertificate, Signatory +from credentials.apps.edx_credentials_extensions.edly_credentials_app.api.fields import Base64ImageField + + +class SignatorySerializer(serializers.ModelSerializer): + """ + Serializer for Signatory. + """ + image = Base64ImageField(max_length=None, use_url=True) + + class Meta: + model = Signatory + fields = ['id', 'name', 'title', 'image'] + + +class ProgramCertificateConfigurationSerializer(serializers.ModelSerializer): + """ + Serializer for Program Certificate Configuration. + """ + site = serializers.SlugRelatedField(slug_field='domain', queryset=Site.objects.all()) + signatories = SignatorySerializer(many=True) + + class Meta: + model = ProgramCertificate + fields = [ + 'site', 'is_active', 'signatories', 'program_uuid', 'use_org_name', + 'include_hours_of_effort', 'language', + ] + + def create(self, validated_data): + signatories = [] + for signatory in validated_data.pop('signatories', []): + signatories.append(Signatory.objects.create(**signatory)) + + program_certificate_configuration = ProgramCertificate(**validated_data) + program_certificate_configuration.save() + program_certificate_configuration.signatories.set(signatories) + return program_certificate_configuration + + def update(self, instance, validated_data): + signatory_ids = self.context.get('signatory_ids', []) + + signatories = [] + for signatory_index, signatory in enumerate(validated_data.pop('signatories', [])): + signatory_obj, _ = Signatory.objects.update_or_create( + pk=signatory_ids[signatory_index], + defaults=signatory + ) + signatories.append(signatory_obj) + + for attr, value in validated_data.items(): + setattr(instance, attr, value) + + instance.save() + instance.signatories.set(signatories) + return instance diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/tests/__init__.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/tests/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/tests/test_serializers.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/tests/test_serializers.py new file mode 100644 index 000000000..80312a250 --- /dev/null +++ b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/tests/test_serializers.py @@ -0,0 +1,85 @@ +""" +Tests for edly_api serializers. +""" +from django.test import TestCase + +from credentials.apps.catalog.tests.factories import ProgramFactory +from credentials.apps.core.tests.mixins import SiteMixin +from credentials.apps.credentials.tests.factories import SignatoryFactory, ProgramCertificateFactory +from credentials.apps.edx_credentials_extensions.edly_credentials_app.api.serializers import SignatorySerializer, ProgramCertificateConfigurationSerializer + + +class SignatorySerializerTest(SiteMixin, TestCase): + """ + Tests for Signatory Serializer. + """ + + def setUp(self): + super(SignatorySerializerTest, self).setUp() + self.serializer = SignatorySerializer + self.test_data = { + "name": "Jhon Doe", + "title": "Director Certificates", + "image": "" + } + + def test_create_signatory(self): + """ + Verify creation data. + """ + serializer = self.serializer(data=self.test_data) + assert serializer.is_valid(raise_exception=True) + + def test_update_signatory(self): + """ + Verify update data. + """ + signatory = SignatoryFactory() + self.test_data['id'] = signatory.id + serializer = self.serializer(data=self.test_data) + assert serializer.is_valid(raise_exception=True) + + +class ProgramCertificateConfigurationSerializerTest(SiteMixin, TestCase): + """ + Tests for Program Certificate Configuration Serializer. + """ + + def setUp(self): + super(ProgramCertificateConfigurationSerializerTest, self).setUp() + self.serializer = ProgramCertificateConfigurationSerializer + + program = ProgramFactory(site=self.site) + self.test_data = { + "site": self.site.domain, + "is_active": True, + "signatories": [ + { + "name": "Jhon Doe", + "title": "Director Certificates", + "image": "" + } + ], + "program_uuid": str(program.uuid), + "use_org_name": True, + "include_hours_of_effort": True, + "language": "en" + } + + def test_create_program_certificate_configuration(self): + """ + Verify creation data. + """ + serializer = self.serializer(data=self.test_data) + assert serializer.is_valid(raise_exception=True) + + def test_update_program_certificate_configuration(self): + """ + Verify update data. + """ + test_data = self.test_data + test_data['signatories'] = [] + test_data['is_active'] = False + + serializer = self.serializer(data=test_data) + assert serializer.is_valid(raise_exception=True) diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/urls.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/urls.py new file mode 100644 index 000000000..a9988e30a --- /dev/null +++ b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/urls.py @@ -0,0 +1,5 @@ +from django.conf.urls import include, url + +urlpatterns = [ + url(r'^v1/', include('credentials.apps.edx_credentials_extensions.edly_credentials_app.api.v1.urls')), +] diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/__init__.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/urls.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/urls.py new file mode 100644 index 000000000..74ee2ffae --- /dev/null +++ b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/urls.py @@ -0,0 +1,9 @@ +from rest_framework import routers + +from credentials.apps.edx_credentials_extensions.edly_credentials_app.api.v1.views.program_certificate_configuration import ProgramCertificateConfigurationViewSet + + +router = routers.SimpleRouter() +router.register(r'program-certificate-configuration', ProgramCertificateConfigurationViewSet, base_name='program-certificate-configuration') + +urlpatterns = router.urls diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/views/__init__.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/views/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/views/program_certificate_configuration.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/views/program_certificate_configuration.py new file mode 100644 index 000000000..ff81c78c0 --- /dev/null +++ b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/views/program_certificate_configuration.py @@ -0,0 +1,46 @@ +""" +Views for Edly Program Certificate Configuration API. +""" +from rest_framework import status, viewsets +from rest_framework.permissions import IsAuthenticated +from rest_framework.response import Response + +from credentials.apps.credentials.models import ProgramCertificate +from credentials.apps.edx_credentials_extensions.edly_credentials_app.api.permissions import CanAccessCurrentSite +from credentials.apps.edx_credentials_extensions.edly_credentials_app.api.serializers import ( + ProgramCertificateConfigurationSerializer +) + + +class ProgramCertificateConfigurationViewSet(viewsets.ModelViewSet): + """ + ViewSet for Program Certificate Configuration. + """ + lookup_field = 'program_uuid' + lookup_value_regex = '[0-9a-f-]+' + permission_classes = (IsAuthenticated, CanAccessCurrentSite) + serializer_class = ProgramCertificateConfigurationSerializer + queryset = ProgramCertificate.objects.all() + + def update(self, request, *args, **kwargs): + """ + Update existing Program Certificate Configuration. + """ + data = request.data.copy() + + signatory_ids = [] + for signatory in data.get('signatories', []): + signatory_ids.append(signatory.get('id')) + + context = { + 'request': request, + 'signatory_ids': signatory_ids + } + + instance = ProgramCertificate.objects.get(program_uuid=kwargs.get('program_uuid')) + + serializer = ProgramCertificateConfigurationSerializer(instance, data=data, context=context) + serializer.is_valid(raise_exception=True) + self.perform_update(serializer) + headers = self.get_success_headers(serializer.data) + return Response(serializer.data, status=status.HTTP_200_OK, headers=headers) diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/views/tests/__init__.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/views/tests/__init__.py new file mode 100644 index 000000000..e69de29bb diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/views/tests/test_views.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/views/tests/test_views.py new file mode 100644 index 000000000..cc998aa89 --- /dev/null +++ b/credentials/apps/edx_credentials_extensions/edly_credentials_app/api/v1/views/tests/test_views.py @@ -0,0 +1,101 @@ +""" +Tests for edly_api Views. +""" +import json + +from django.urls import reverse + +from rest_framework.test import APITestCase + +from credentials.apps.catalog.tests.factories import ProgramFactory +from credentials.apps.core.tests.factories import UserFactory +from credentials.apps.core.tests.mixins import SiteMixin +from credentials.apps.credentials.tests.factories import ProgramCertificateFactory + +JSON_CONTENT_TYPE = 'application/json' + + +def get_test_image_as_base_64_encoded_string(): + base64_header = 'data:image/png;base64,' + base64_data = ( + 'iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAQAAAC1HAwCAAAAC0lEQVR42mNk+A8AAQUBAScY' + '42YAAAAASUVORK5CYII=' + ) + base64_full = base64_header + base64_data + return base64_full + + +class ProgramCertificateConfigurationTests(SiteMixin, APITestCase): + """ + Test for Program Certificate Configuration Viewset. + """ + + def setUp(self): + super(ProgramCertificateConfigurationTests, self).setUp() + self.list_url = reverse('edly_api:program-certificate-configuration-list') + + self.user = UserFactory(is_staff=True, is_superuser=True) + + program = ProgramFactory(site=self.site) + self.test_data = { + 'site': self.site.domain, + 'is_active': True, + 'signatories': [ + { + 'name': 'Jhon Doe', + 'title': 'Test Certificate', + 'image': get_test_image_as_base_64_encoded_string() + } + ], + 'program_uuid': str(program.uuid), + 'use_org_name': True, + 'include_hours_of_effort': True, + 'language': 'en' + } + + def test_create_program_certificate_configuration(self): + """ + Verify program certificate configuration creation. + """ + response = self.client.post(self.list_url, data=json.dumps(self.test_data), content_type=JSON_CONTENT_TYPE) + assert response.status_code == 401 + + self.client.login(username=self.user.username, password='password') + response = self.client.post(self.list_url, data=json.dumps(self.test_data), content_type=JSON_CONTENT_TYPE) + assert response.status_code == 201 + + def test_update_program_certificate_configuration(self): + """ + Verify program certificate configuration update. + """ + program_certificate_configuration = ProgramCertificateFactory() + + self.detail_url = reverse('edly_api:program-certificate-configuration-detail', + kwargs={'program_uuid': str(program_certificate_configuration.program_uuid)}) + test_data = self.test_data + test_data['signatories'] = [] + test_data['is_active'] = False + + response = self.client.patch(self.detail_url, data=json.dumps(test_data), content_type=JSON_CONTENT_TYPE) + assert response.status_code == 401 + + self.client.login(username=self.user.username, password='password') + response = self.client.patch(self.detail_url, data=json.dumps(test_data), content_type=JSON_CONTENT_TYPE) + assert response.status_code == 200 + + def test_list_program_certificate_configuration(self): + """ + Verify list program certificate configuration. + """ + program_certificate_configuration = ProgramCertificateFactory() + + response = self.client.get(self.list_url) + assert response.status_code == 401 + + self.client.login(username=self.user.username, password='password') + response = self.client.get(self.list_url) + assert response.status_code == 200 + + actual_data = response.json() + actual_data = actual_data.get('results')[0] + assert str(program_certificate_configuration.program_uuid) == actual_data.get('program_uuid') diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/tests/test_utils.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/tests/test_utils.py new file mode 100644 index 000000000..bb94cd7bd --- /dev/null +++ b/credentials/apps/edx_credentials_extensions/edly_credentials_app/tests/test_utils.py @@ -0,0 +1,116 @@ +""" +Tests for Edly Utils Functions. +""" +from unittest import TestCase + +import jwt +import mock +import pytest +from django.conf import settings +from django.test.client import RequestFactory +from mock import MagicMock + +from credentials.apps.core.tests.factories import SiteFactory, UserFactory +from credentials.apps.edx_credentials_extensions.edly_credentials_app.tests.factories import ( + EdlySiteConfigurationFactory, +) +from credentials.apps.edx_credentials_extensions.edly_credentials_app.utils import ( + encode_edly_user_info_cookie, + decode_edly_user_info_cookie, + get_edx_org_from_cookie, + user_has_edx_organization_access, +) + + +@pytest.mark.django_db +class UtilsTests(TestCase): + """ + Tests for utility methods. + """ + + def setUp(self): + """ + Setup initial test data + """ + super(UtilsTests, self).setUp() + self.user = UserFactory() + self.admin_user = UserFactory(is_staff=True) + self.request = RequestFactory().get('/') + self.request.user = self.user + self.request.session = self._get_stub_session() + self.request.site = SiteFactory() + self.request.site.siteconfiguration = EdlySiteConfigurationFactory( + site=self.request.site, + edx_org_short_name='cloudX' + ) + + self.test_edly_user_info_cookie_data = { + 'edly-org': 'edly', + 'edly-sub-org': 'cloud', + 'edx-org': 'cloudX' + } + + def _get_stub_session(self, expire_at_browser_close=False, max_age=604800): + return MagicMock( + get_expire_at_browser_close=lambda: expire_at_browser_close, + get_expiry_age=lambda: max_age, + ) + + def _copy_cookies_to_request(self, request): + edly_cookie_string = encode_edly_user_info_cookie((self.test_edly_user_info_cookie_data)) + request.COOKIES = { + settings.EDLY_USER_INFO_COOKIE_NAME: edly_cookie_string + } + + def test_encode_edly_user_info_cookie(self): + """ + Test that "encode_edly_user_info_cookie" method encodes data correctly. + """ + actual_encoded_string = encode_edly_user_info_cookie(self.test_edly_user_info_cookie_data) + expected_encoded_string = jwt.encode( + self.test_edly_user_info_cookie_data, settings.EDLY_COOKIE_SECRET_KEY, + algorithm=settings.EDLY_JWT_ALGORITHM + ) + + assert actual_encoded_string == expected_encoded_string + + def test_decode_edly_user_info_cookie(self): + """ + Test that "decode_edly_user_info_cookie" method decodes data correctly. + """ + encoded_data = jwt.encode( + self.test_edly_user_info_cookie_data, + settings.EDLY_COOKIE_SECRET_KEY, + algorithm=settings.EDLY_JWT_ALGORITHM + ) + decoded_edly_user_info_cookie_data = decode_edly_user_info_cookie(encoded_data) + + assert self.test_edly_user_info_cookie_data == decoded_edly_user_info_cookie_data + + def test_get_edx_org_from_cookie(self): + """ + Test that "get_edx_org_from_cookie" method returns edx-org short name correctly. + """ + edly_user_info_cookie = encode_edly_user_info_cookie(self.test_edly_user_info_cookie_data) + + assert self.request.site.siteconfiguration.edx_org_short_name == get_edx_org_from_cookie(edly_user_info_cookie) + + def test_user_with_edx_organization_access(self): + """ + Test user has edx organization access of a site. + """ + with mock.patch('credentials.apps.edx_credentials_extensions.edly_credentials_app.utils.get_current_site') as current_site: + current_site.return_value = self.request.site + self._copy_cookies_to_request(self.request) + + assert user_has_edx_organization_access(self.request) is True + + def test_user_without_edx_organization_access(self): + """ + Test user does not have edx organization access of a site. + """ + with mock.patch('credentials.apps.edx_credentials_extensions.edly_credentials_app.utils.get_current_site') as current_site: + current_site.return_value = self.request.site + self.request.site.siteconfiguration = EdlySiteConfigurationFactory() + + assert user_has_edx_organization_access(self.request) is False diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/urls.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/urls.py index 65b34d128..4f3dd04a2 100644 --- a/credentials/apps/edx_credentials_extensions/edly_credentials_app/urls.py +++ b/credentials/apps/edx_credentials_extensions/edly_credentials_app/urls.py @@ -1,6 +1,5 @@ -from django.conf.urls import url - -from . import views +from django.conf.urls import include, url urlpatterns = [ + url('', include('credentials.apps.edx_credentials_extensions.edly_credentials_app.api.urls')), ] diff --git a/credentials/apps/edx_credentials_extensions/edly_credentials_app/utils.py b/credentials/apps/edx_credentials_extensions/edly_credentials_app/utils.py new file mode 100644 index 000000000..3d19aa90f --- /dev/null +++ b/credentials/apps/edx_credentials_extensions/edly_credentials_app/utils.py @@ -0,0 +1,84 @@ +import logging + +import jwt +from django.conf import settings +from django.contrib.sites.shortcuts import get_current_site + +from credentials.apps.core.models import SiteConfiguration + +LOGGER = logging.getLogger(__name__) + + +def user_has_edx_organization_access(request): + """ + Check if the user can access the requested URL by verifying its linked edx organization. + + This method checks if the "edx_org_short_name" related to the requested URL's site + matches with the "edx-org" in the "edly_user_info" cookie. + + Arguments: + request: HTTP request object + + Returns: + bool: Returns True if user has edX organization access otherwise False. + """ + + if request.user.is_superuser or request.user.is_staff: + return True + + current_site = get_current_site(request) + try: + current_site_configuration = current_site.siteconfiguration + except SiteConfiguration.DoesNotExist: + LOGGER.warning('Site (%s) has no related site configuration.', current_site) + return False + + edly_user_info_cookie = request.COOKIES.get(settings.EDLY_USER_INFO_COOKIE_NAME, None) + if current_site_configuration.edx_org_short_name == get_edx_org_from_cookie(edly_user_info_cookie): + return True + + return False + + +def encode_edly_user_info_cookie(cookie_data): + """ + Encode edly_user_info cookie data into JWT string. + + Arguments: + cookie_data (dict): Edly user info cookie dict. + + Returns: + string + """ + return jwt.encode(cookie_data, settings.EDLY_COOKIE_SECRET_KEY, algorithm=settings.EDLY_JWT_ALGORITHM) + + +def decode_edly_user_info_cookie(encoded_cookie_data): + """ + Decode edly_user_info cookie data from JWT string. + + Arguments: + encoded_cookie_data (dict): Edly user info cookie JWT encoded string. + + Returns: + dict + """ + return jwt.decode(encoded_cookie_data, settings.EDLY_COOKIE_SECRET_KEY, algorithms=[settings.EDLY_JWT_ALGORITHM]) + + +def get_edx_org_from_cookie(encoded_cookie_data): + """ + Returns edx-org short name from the edly-user-info cookie. + + Arguments: + encoded_cookie_data (dict): Edly user info cookie JWT encoded string. + + Returns: + string + """ + + if not encoded_cookie_data: + return '' + + decoded_cookie_data = decode_edly_user_info_cookie(encoded_cookie_data) + return decoded_cookie_data['edx-org'] diff --git a/credentials/settings/base.py b/credentials/settings/base.py index 7aedf92cd..4fae96e73 100644 --- a/credentials/settings/base.py +++ b/credentials/settings/base.py @@ -66,6 +66,7 @@ 'credentials.apps.edx_django_extensions', 'credentials.apps.credentials_theme_openedx', 'credentials.apps.records', + 'credentials.apps.edx_credentials_extensions.edly_credentials_app' ] INSTALLED_APPS += THIRD_PARTY_APPS @@ -434,3 +435,8 @@ 'debug_toolbar.panels.redirects.RedirectsPanel', ] # END DJANGO DEBUG TOOLBAR CONFIGURATION + +# Edly user info cookie specific values +EDLY_USER_INFO_COOKIE_NAME = 'edly-user-info' +EDLY_COOKIE_SECRET_KEY = 'EDLY-COOKIE-SECRET-KEY' +EDLY_JWT_ALGORITHM = 'HS256' diff --git a/credentials/settings/production.py b/credentials/settings/production.py index 2b0dc7fcd..dde3118f3 100644 --- a/credentials/settings/production.py +++ b/credentials/settings/production.py @@ -58,3 +58,8 @@ for override, value in DB_OVERRIDES.items(): DATABASES['default'][override] = value + +# Edly user info cookie specific values +EDLY_USER_INFO_COOKIE_NAME = environ.get('EDLY_USER_INFO_COOKIE_NAME', EDLY_USER_INFO_COOKIE_NAME) +EDLY_COOKIE_SECRET_KEY = environ.get('EDLY_COOKIE_SECRET_KEY', EDLY_COOKIE_SECRET_KEY) +EDLY_JWT_ALGORITHM = environ.get('EDLY_JWT_ALGORITHM', EDLY_JWT_ALGORITHM) diff --git a/credentials/urls.py b/credentials/urls.py index 70cdd4c6f..fbfc18cfd 100644 --- a/credentials/urls.py +++ b/credentials/urls.py @@ -48,6 +48,11 @@ url(r'^hijack/', include('hijack.urls', namespace='hijack')), ] +# Edly Urls +urlpatterns += [ + url(r'^edly-api/', include('credentials.apps.edx_credentials_extensions.edly_credentials_app.urls', namespace='edly_api')), +] + handler500 = 'credentials.apps.core.views.render_500' # Add media and extra urls