Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Check for LMS ID first when authenticating from LMS #2645

Merged
merged 3 commits into from
Jan 16, 2025

Conversation

Alex-Jordan
Copy link
Contributor

This is an attempt to deal with situations where a user in the LMS undergoes a name change. So now when then try to authenticate form the LMS, we first look at their LMS ID and see if there is a WW user with that same lis_source_did.

For now I have only made changes for LTI 1.3. But it turns out I can't test this at all for LTI 1.3 because the WW server I have for testing LTI with our D2L test server is not set up for https. So I can only test LTI 1.1 stuff.

Any feedback on this code so far?

@Alex-Jordan Alex-Jordan added the Do Not Merge Yet PR to allow others to inspect -- not ready for prime time label Dec 17, 2024
@Alex-Jordan Alex-Jordan marked this pull request as draft December 17, 2024 20:50
@Alex-Jordan Alex-Jordan removed the Do Not Merge Yet PR to allow others to inspect -- not ready for prime time label Dec 17, 2024
}
if ($user) {
$user_id_source = $c->stash->{lti_lms_user_id};
$type_of_source = 'lis_source_did';
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This really shouldn't be called the lis_source_did. That isn't what it is. It is the sub claim which is a stable locally unique identifier for the user in the LMS. See https://www.imsglobal.org/spec/lti/v1p3#message-claims. Nowhere in the LTI 1.3 specification do they even refer to an lis_sourced_id. That is a LTI 1.1 thing.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think that the $type_of_source should be "database user" in this case. Note that this is only used for the debugging message on line 217 below, so it can contain spaces and be informative. The type of source means the source of the webwork2 user_id. That is from the user_id column of the user table in the database in this case, and not the misnamed lis_source_did column.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Okay, looking at this closer I also see that $user_id_source = $c->stash->{lti_lms_user_id} is not good. The values of $user_id_source and $type_of_source need to be set to something that will work well in the sentence "User id is |$self->{user_id}| (obtained from $user_id_source which was $type_of_source)\n". That is all that those variables are used for (other than the usage of $user_id_source in the case that it is equal to "email").

To be honest, I don't think anything will work in that sentence well in this case. So instead I think the $type_of_source should just be set to "database user" in this case, and "$user_id_source which was ..." in the other two cases. Then the debugging message changed to "User id is |$self->{user_id}| (obtained from $type_of_source)\n";

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So, what I was doing here was the best I could think of for User id is ____ (obtained from ____ which was ____). Limiting to that structure, User id is ____ (obtained from lis_source_did which was xxx-yyy-zzz) seemed OK, assuming someone reading the log would know what that means. But having a different sentence altogether is better.

}
# First check if we already have a user with the current lis_source_did
my $user = ($c->db->getUsersWhere({ lis_source_did => $c->stash->{lti_lms_user_id} }))[0] // ''
if $c->stash->{lti_lms_user_id};
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The $c->stash->{lti_lms_user_id} can never be set at this point. It isn't set until line 236 later in this same method. You will need to use $claims->{sub} at this point instead.

@@ -188,6 +204,8 @@ sub get_credentials ($self) {
[ recitation => 'https://purl.imsglobal.org/spec/lti/claim/custom#recitation' ],
);

$self->{lis_source_did} = $c->stash->{lti_lms_user_id} if $c->stash->{lti_lms_user_id};
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why is this set? It will never be needed.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This was my attempt at recording the LMS ID as a new user is created. I didn't track the code, but I assumed the previous block was setting similar things like first_name, and that be setting this here it would trickle down to the lis_source_did column being filled out for the new user. I'm not surprised if that's wrong, but I couldn't test.

@drgrice1
Copy link
Member

I added a pull request to this branch with some suggested changes. I tested it with Moodle, and it works. I created a user, and signed in. Then I changed the user's email address in the LMS (I have $LTI{v1p3}{preferred_source_of_username} = 'email'), and then signed in again, and the user was signed in with the original webwork2 user. No new user was created.

@drgrice1
Copy link
Member

I have now also successfully tested my suggested changes with Canvas.

@Alex-Jordan
Copy link
Contributor Author

Thanks for looking at this, and I'll merge your PR soon. I hope to get this in my PCC production server for our winter term. For now at PCC, doing it with LTI 1.3 is enough.

Suggestions to make using an existing user with given LMS id work.
@Alex-Jordan
Copy link
Contributor Author

I merged your PR, thanks again. I may take a stab at the parallel development for LTI 1.1.

@drgrice1
Copy link
Member

Unfortunately, I don't think this is going to work for LTI 1.1. At least not without adding a new column to the user table in the database.

The thing is that the lis_sourced_id is not an LMS user identifier. In fact it is a resource identifier for the external tool link (or grade item really) in the LMS. That is saved in the lis_source_did column for the user in the case that grade pass back mode is "course", but it would not be reliable to use it for this even in that case. It simply isn't a user identifier. Looking at what Moodle and Canvas send for this in particular it would be a bad idea to use. Moodle sends something like {"data":{"instanceid":"62","userid":"4","typeid":"1","launchid":69624527},"hash":"c0284102b6441f01028d784af97c9695b61e181852b18d0ceb155f1b767dfc9e"} for this. Canvas sends something like 2712-36676-392282-120885-1b74b9b323c2b11a439870b54f6f3ebac1bc6ceb.

Looking at the LTI 1.1 specification (see section 3 on basic launch data in https://www.imsglobal.org/specs/ltiv1p1p1/implementation-guide) and what Moodle and Canvas send, it seems the only reliable thing to use would be the user_id parameter. In fact that is literally the only thing in common between what Moodle and Canvas send that could possibly be used for this. The specification doesn't technically require that parameter, it is only recommended, but both Moodle and Canvas send it, and it is in fact the same thing that they send for the sub claim for LTI 1.3. The problem is that you can't save that in the lis_source_did column of the user table in the database because that would conflict with the value that is needed there for grade pass back in the case that the grade pass back mode is "course". I suppose if the grade pass back mode is unset or is "homework" you could, but I don't think that would be a good idea. Certainly the capability of webwork2 to support this shouldn't depend on the grade pass back mode.

@Alex-Jordan
Copy link
Contributor Author

OK, then I guess it's time to lift the "draft" status from this?

@Alex-Jordan Alex-Jordan marked this pull request as ready for review December 18, 2024 22:23
@drgrice1
Copy link
Member

I think it is.

@Alex-Jordan
Copy link
Contributor Author

I applied this to my production server and ran the following test with D2L. Working directly with the database, for one WW course, I removed my user's lis_source_did, and I set a different user to have my lis_source_did. Then I entered the WW course using an LTI link from D2L, and it entered me as that other user. In other words, the test was a success.

I have not tested the part of this that records the lis_source_did when a new user is created.

Copy link
Member

@drgrice1 drgrice1 left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Now that this is not a draft anymore, I will add my approval. I have already tested it with Moodle and Canvas as stated previously.

@somiaj
Copy link
Contributor

somiaj commented Dec 19, 2024

I can test this, but it looks good so I can throw my approval. I'm assuming you both have also tested that creation of new users still works with this?

@drgrice1 you comment that this won't be reliable for LTI 1.1, because what is being saved is not a user id but more a resource id. But how often would this actually change for a single user? And I assume any change could never match a different existing user? So could we use it to try and catch name changes in the case the LMS doesn't update this resource id for the updated user?

@drgrice1
Copy link
Member

If you are using grade pass back mode, then it will change for every set you log in with, and there is no guarantee it won't also work for another user. It simply will not work, and trying to use it for this would be completely wrong in any case.

Also, yes I have tested it with user creation.

@Alex-Jordan
Copy link
Contributor Author

I just tested with account creation too (with D2L). It worked. And checking the database, the account creation was enough to get the lis_source_did in there without any attempt at grade passback.

@somiaj
Copy link
Contributor

somiaj commented Dec 19, 2024

Oh nevermind, I see what is going on now, didn't fully understand your previous statement.

One thought if we wanted to try to support this in 1.1, could we just make the lis_source_did contain both the user_id and the resource_id, something like user_id;lis_source_did. It would make things more complicated but could allow this change without another column in the DB. Though maybe another column would be better for this instead of just stashing the 1.3 user_id in the 1.1 resource_id column, then we could support both 1.3 and 1.1 with this feature.

@somiaj
Copy link
Contributor

somiaj commented Dec 19, 2024

Anyways here is my approval, I was just trying to think of ways to support 1.1 because I'm a dinosaur who doesn't want to deal with the headache of moving to 1.3 here. Though I guess 1.1 probably going to be dropped in the future at some point anyways by the LMSes?

@Alex-Jordan
Copy link
Contributor Author

My opinion: it would be OK to start treating LTI 1.1 as basically frozen.

@dlglin
Copy link
Member

dlglin commented Dec 19, 2024

My opinion: it would be OK to start treating LTI 1.1 as basically frozen.

LTI 1.1 is not officially deprecated, but hasn't been supported since June 30, 2021: https://www.1edtech.org/lti-security-announcement-and-deprecation-schedule

I'm fine with not adding new functionality to LTI 1.1, especially since some new features might not be possible in 1.1.

@somiaj somiaj merged commit fe3e59f into openwebwork:develop Jan 16, 2025
2 checks passed
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants