-
Notifications
You must be signed in to change notification settings - Fork 3.9k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: link LTI Provider launches to authenticated users
With this change, the platform users who access content via LTI will be automatically linked to their platform account instead of the new (anonymous) one. The following conditions need to be met: * The `LtiConsumer` should be configured to auto-link the users via email. * The LTI Consumer should share the user's email using the `lis_person_contact_email_primary` parameter in the LTI Launch POST data. This also replaces the one-to-one relationship of the `User` and `LtiUser` with one-to-many. This way, multiple `LtiUser` objects can refer to the same `edx_user`. With the auto-linking, multiple LTI Consumers can create independent `LtiUser` objects with the same `edx_user`. Co-authored-by: Piotr Surowiec <[email protected]>
- Loading branch information
1 parent
7050eca
commit 5b2f012
Showing
7 changed files
with
227 additions
and
34 deletions.
There are no files selected for viewing
26 changes: 26 additions & 0 deletions
26
lms/djangoapps/lti_provider/migrations/0004_require_user_account.py
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
# Generated by Django 3.2.22 on 2023-11-06 09:47 | ||
|
||
from django.conf import settings | ||
from django.db import migrations, models | ||
import django.db.models.deletion | ||
|
||
|
||
class Migration(migrations.Migration): | ||
|
||
dependencies = [ | ||
migrations.swappable_dependency(settings.AUTH_USER_MODEL), | ||
('lti_provider', '0003_auto_20161118_1040'), | ||
] | ||
|
||
operations = [ | ||
migrations.AddField( | ||
model_name='lticonsumer', | ||
name='require_user_account', | ||
field=models.BooleanField(blank=True, default=False, help_text='When checked, the LTI content will load only for learners who have an account in this instance. This is required only for linking learner accounts with the LTI consumer. See the Open edX LTI Provider documentation for more details.'), | ||
), | ||
migrations.AlterField( | ||
model_name='ltiuser', | ||
name='edx_user', | ||
field=models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to=settings.AUTH_USER_MODEL), | ||
), | ||
] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -7,8 +7,9 @@ | |
from unittest.mock import MagicMock, PropertyMock, patch | ||
|
||
import pytest | ||
from django.contrib.auth.models import User # lint-amnesty, pylint: disable=imported-auth-user | ||
from django.contrib.auth.models import AnonymousUser, User # lint-amnesty, pylint: disable=imported-auth-user | ||
from django.core.exceptions import PermissionDenied | ||
from django.db.utils import IntegrityError | ||
from django.test import TestCase | ||
from django.test.client import RequestFactory | ||
|
||
|
@@ -92,15 +93,22 @@ def setUp(self): | |
self.old_user = UserFactory.create() | ||
self.request = RequestFactory().post('/') | ||
self.request.user = self.old_user | ||
self.auto_linking_consumer = LtiConsumer( | ||
consumer_name='AutoLinkingConsumer', | ||
consumer_key='AutoLinkingKey', | ||
consumer_secret='AutoLinkingSecret', | ||
require_user_account=True | ||
) | ||
self.auto_linking_consumer.save() | ||
|
||
def create_lti_user_model(self): | ||
def create_lti_user_model(self, consumer=None): | ||
""" | ||
Generate and save a User and an LTI user model | ||
""" | ||
edx_user = User(username=self.edx_user_id) | ||
edx_user.save() | ||
lti_user = LtiUser( | ||
lti_consumer=self.lti_consumer, | ||
lti_consumer=consumer or self.lti_consumer, | ||
lti_user_id=self.lti_user_id, | ||
edx_user=edx_user | ||
) | ||
|
@@ -140,6 +148,38 @@ def test_authentication_with_wrong_user(self, create_user, switch_user): | |
assert not create_user.called | ||
switch_user.assert_called_with(self.request, lti_user, self.lti_consumer) | ||
|
||
def test_auto_linking_of_users_using_lis_person_contact_email_primary(self, create_user, switch_user): | ||
request = RequestFactory().post("/", {"lis_person_contact_email_primary": self.old_user.email}) | ||
request.user = self.old_user | ||
|
||
users.authenticate_lti_user(request, self.lti_user_id, self.lti_consumer) | ||
create_user.assert_called_with(self.lti_user_id, self.lti_consumer) | ||
|
||
users.authenticate_lti_user(request, self.lti_user_id, self.auto_linking_consumer) | ||
create_user.assert_called_with(self.lti_user_id, self.auto_linking_consumer, self.old_user.email) | ||
|
||
def test_raise_exception_trying_to_auto_link_unauthenticate_user(self, create_user, switch_user): | ||
request = RequestFactory().post("/") | ||
request.user = AnonymousUser() | ||
|
||
with self.assertRaises(PermissionDenied): | ||
users.authenticate_lti_user(request, self.lti_user_id, self.auto_linking_consumer) | ||
|
||
def test_raise_exception_on_mismatched_user_and_lis_email(self, create_user, switch_user): | ||
request = RequestFactory().post("/", {"lis_person_contact_email_primary": "[email protected]"}) | ||
request.user = self.old_user | ||
|
||
with self.assertRaises(PermissionDenied): | ||
users.authenticate_lti_user(request, self.lti_user_id, self.auto_linking_consumer) | ||
|
||
def test_authenticate_unauthenticated_user_after_auto_linking_of_user_account(self, create_user, switch_user): | ||
lti_user = self.create_lti_user_model(self.auto_linking_consumer) | ||
self.request.user = AnonymousUser() | ||
|
||
users.authenticate_lti_user(self.request, self.lti_user_id, self.auto_linking_consumer) | ||
assert not create_user.called | ||
switch_user.assert_called_with(self.request, lti_user, self.auto_linking_consumer) | ||
|
||
|
||
class CreateLtiUserTest(TestCase): | ||
""" | ||
|
@@ -154,16 +194,17 @@ def setUp(self): | |
consumer_secret='TestSecret' | ||
) | ||
self.lti_consumer.save() | ||
self.existing_user = UserFactory.create() | ||
|
||
def test_create_lti_user_creates_auth_user_model(self): | ||
users.create_lti_user('lti_user_id', self.lti_consumer) | ||
assert User.objects.count() == 1 | ||
assert User.objects.count() == 2 | ||
|
||
@patch('uuid.uuid4', return_value='random_uuid') | ||
@patch('lms.djangoapps.lti_provider.users.generate_random_edx_username', return_value='edx_id') | ||
def test_create_lti_user_creates_correct_user(self, uuid_mock, _username_mock): | ||
users.create_lti_user('lti_user_id', self.lti_consumer) | ||
assert User.objects.count() == 1 | ||
assert User.objects.count() == 2 | ||
user = User.objects.get(username='edx_id') | ||
assert user.email == '[email protected]' | ||
uuid_mock.assert_called_with() | ||
|
@@ -173,10 +214,34 @@ def test_unique_username_created(self, username_mock): | |
User(username='edx_id').save() | ||
users.create_lti_user('lti_user_id', self.lti_consumer) | ||
assert username_mock.call_count == 2 | ||
assert User.objects.count() == 2 | ||
assert User.objects.count() == 3 | ||
user = User.objects.get(username='new_edx_id') | ||
assert user.email == '[email protected]' | ||
|
||
def test_existing_user_is_linked(self): | ||
lti_user = users.create_lti_user('lti_user_id', self.lti_consumer, self.existing_user.email) | ||
assert lti_user.lti_consumer == self.lti_consumer | ||
assert lti_user.edx_user == self.existing_user | ||
|
||
def test_only_one_lti_user_edx_user_for_each_lti_consumer(self): | ||
users.create_lti_user('lti_user_id', self.lti_consumer, self.existing_user.email) | ||
|
||
with pytest.raises(IntegrityError): | ||
users.create_lti_user('lti_user_id', self.lti_consumer, self.existing_user.email) | ||
|
||
def test_create_multiple_lti_users_for_edx_user_if_lti_consumer_varies(self): | ||
lti_consumer_2 = LtiConsumer( | ||
consumer_name="SecondConsumer", | ||
consumer_key="SecondKey", | ||
consumer_secret="SecondSecret", | ||
) | ||
lti_consumer_2.save() | ||
|
||
lti_user_1 = users.create_lti_user('lti_user_id', self.lti_consumer, self.existing_user.email) | ||
lti_user_2 = users.create_lti_user('lti_user_id', lti_consumer_2, self.existing_user.email) | ||
|
||
assert lti_user_1.edx_user == lti_user_2.edx_user | ||
|
||
|
||
class LtiBackendTest(TestCase): | ||
""" | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -5,6 +5,7 @@ | |
|
||
from unittest.mock import MagicMock, patch | ||
|
||
from django.contrib.auth.models import AnonymousUser | ||
from django.test import TestCase | ||
from django.test.client import RequestFactory | ||
from django.urls import reverse | ||
|
@@ -58,7 +59,7 @@ def build_launch_request(extra_post_data=None, param_to_delete=None): | |
del post_data[param_to_delete] | ||
request = RequestFactory().post('/', data=post_data) | ||
request.user = UserFactory.create() | ||
request.session = {} | ||
request.session = MagicMock() | ||
return request | ||
|
||
|
||
|
@@ -82,6 +83,14 @@ def setUp(self): | |
) | ||
self.consumer.save() | ||
|
||
self.auto_link_consumer = models.LtiConsumer( | ||
consumer_name='auto-link-consumer', | ||
consumer_key='consumer_key_2', | ||
consumer_secret='secret_2', | ||
require_user_account=True | ||
) | ||
self.auto_link_consumer.save() | ||
|
||
|
||
class LtiLaunchTest(LtiTestMixin, TestCase): | ||
""" | ||
|
@@ -189,6 +198,36 @@ def test_lti_consumer_record_supplemented_with_guid(self, _render): | |
) | ||
assert consumer.instance_guid == 'consumer instance guid' | ||
|
||
@patch('lms.djangoapps.lti_provider.views.render_to_response') | ||
def test_unauthenticated_user_shown_error_when_require_user_account_is_enabled(self, render_error): | ||
""" | ||
Verify that an error page is shown instead of LTI Content for an unauthenticated user, | ||
when the `require_user_account` flag is enabled for the LTI Consumer. | ||
""" | ||
request = build_launch_request({'oauth_consumer_key': 'consumer_key_2'}) | ||
request.user = AnonymousUser() | ||
|
||
views.lti_launch(request, str(COURSE_KEY), str(USAGE_KEY)) | ||
|
||
render_error.assert_called() | ||
assert render_error.call_args[0][0] == "lti_provider/user-auth-error.html" | ||
|
||
@patch('lms.djangoapps.lti_provider.views.render_to_response') | ||
def test_auth_error_shown_when_lis_email_is_different_from_user_email(self, render_error): | ||
""" | ||
When the `require_user_account` flag is enabled for the LTI Consumer, verify that | ||
an error page is shown instead of LTI Content if the authenticated user's email | ||
doesn't match the `lis_person_contact_email_primary` value from LTI Launch. | ||
""" | ||
# lis email different from logged in user | ||
request = build_launch_request({ | ||
'oauth_consumer_key': 'consumer_key_2', | ||
'lis_person_contact_email_primary': '[email protected]' | ||
}) | ||
|
||
views.lti_launch(request, str(COURSE_KEY), str(USAGE_KEY)) | ||
render_error.assert_called() | ||
|
||
|
||
class LtiLaunchTestRender(LtiTestMixin, RenderXBlockTestMixin, ModuleStoreTestCase): | ||
""" | ||
|
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.