Opened 7 years ago
Closed 7 years ago
#28285 closed Bug (worksforme)
Updating Password Hasher Invalidates Users First Login After Upgrade
Reported by: | Simon Blanchard | Owned by: | nobody |
---|---|---|---|
Component: | contrib.auth | Version: | 1.11 |
Severity: | Normal | Keywords: | |
Cc: | Triage Stage: | Unreviewed | |
Has patch: | no | Needs documentation: | no |
Needs tests: | no | Patch needs improvement: | no |
Easy pickings: | no | UI/UX: | no |
Description (last modified by )
The code below, from contrib.auth.base_user.py
, says "hash upgrade is not a password change"
def check_password(self, raw_password): """ Return a boolean of whether the raw_password was correct. Handles hashing formats behind the scenes. """ def setter(raw_password): self.set_password(raw_password) # Password hash upgrades shouldn't be considered password changes. self._password = None self.save(update_fields=["password"]) return check_password(raw_password, self.password, setter)
but the code in contrib.auth.get_user()
verifies the session by comparing a salted hmac of the password hash to the same thing stored in the session.
If we update the password hash the salted hmac will change. Therefore, the session will no longer be considered valid and will be flushed. We can keep the session alive by updating the HASH_SESSION_KEY
in request.session
e.g. via the update_session_auth_hash()
function. But this is not called by the setter function when the hasher is updated.
So the first time the user logs (with the correct password) following an upgrade, their (valid) session will be flushed and they will be logged out again.
Change History (5)
follow-up: 2 comment:1 by , 7 years ago
Description: | modified (diff) |
---|
comment:2 by , 7 years ago
Replying to Tim Graham:
Here's my understanding of the scenario:
- User X logs in and creates session A
- Number of hash iterations changes in Django due to a software upgrade
- Session A is still valid at this point
I don't think so. Hash changes invalidate the session. Hence the need for the update_session_auth_hash() function
- For whatever reason (using a different browser perhaps), user X logs in and creates session B, triggering a password hash upgrade and an invalidation of session A
I think the hash upgrade already happened in your point 2 above
follow-up: 4 comment:3 by , 7 years ago
I'm not following your scenario. Could you please give your steps to reproduce?
comment:4 by , 7 years ago
Replying to Tim Graham:
I'm not following your scenario. Could you please give your steps to reproduce?
def get_user(request): """ Returns the user model instance associated with the given request session. If no user is retrieved an instance of `AnonymousUser` is returned. """ from .models import AnonymousUser user = None try: user_id = _get_user_session_key(request) backend_path = request.session[BACKEND_SESSION_KEY] except KeyError: pass else: if backend_path in settings.AUTHENTICATION_BACKENDS: backend = load_backend(backend_path) user = backend.get_user(user_id) # Verify the session if hasattr(user, 'get_session_auth_hash'): session_hash = request.session.get(HASH_SESSION_KEY) session_hash_verified = session_hash and constant_time_compare( session_hash, user.get_session_auth_hash() ) if not session_hash_verified: request.session.flush() user = None return user or AnonymousUser()
get_user() verifies the user's session hash is the same as the one in the session. If the users session hash (calculated from the password hash) is different from that stored in the session, the session is flushed.
So the scenario is:
- Upgrade Django
- User logs in
- Session hash is set from original password hash before the upgrade
- Django updates password hash but does not update the session hash
- On next request, get_user() notices the session hash and password hash are different
- Session is flushed
- User is logged out
At least that's what I think is happening. But looking at the code I'm not so sure. authenticate() should be callled before login(), authenticate() updates the hash, then login() should be setting the session key with the new hash, oh I may be passing an old user object - with former password hash to login. Sorry my mistake! Thanks!
comment:5 by , 7 years ago
Resolution: | → worksforme |
---|---|
Status: | new → closed |
I can't reproduce a problem with the steps provided, so I'll assume the problem is in your application unless you provide more details.
Here's my understanding of the scenario:
I don't think there's a way to keep session A valid is this scenario. Am I misunderstanding something?