Source code for swh.auth.django.middlewares
# Copyright (C) 2020-2021 The Software Heritage developers
# See the AUTHORS file at the top-level directory of this distribution
# License: GNU Affero General Public License version 3, or any later version
# See top-level LICENSE file for more information
from django.conf import settings
from django.contrib.auth import BACKEND_SESSION_KEY
from django.http.response import HttpResponseRedirect
from swh.auth.django.utils import reverse
[docs]
class OIDCSessionExpiredMiddleware:
"""
Middleware for checking OpenID Connect user session expiration.
That middleware detects when a user previously logged in using
the OpenID Connect authentication backend got his session expired.
In that case it will perform a redirection to a django view whose
name must be set in the ``SWH_AUTH_SESSION_EXPIRED_REDIRECT_VIEW``
django setting (typically a logout view).
The following query parameter will be set for that view:
* ``next``: requested URL before the detection of the session expiration
* ``remote_user``: indicates that the user was previously authenticated with OIDC
"""
def __init__(self, get_response=None):
self.get_response = get_response
self.redirect_view = getattr(
settings, "SWH_AUTH_SESSION_EXPIRED_REDIRECT_VIEW", None
)
if self.redirect_view is None:
raise ValueError(
"SWH_AUTH_SESSION_EXPIRED_REDIRECT_VIEW django setting "
"is mandatory to instantiate OIDCSessionExpiredMiddleware class"
)
self.exempted_urls = [
reverse(v)
for v in (
self.redirect_view,
"oidc-login",
"oidc-login-complete",
"oidc-logout",
)
]
def __call__(self, request):
if (
request.method != "GET"
or request.user.is_authenticated
or BACKEND_SESSION_KEY not in request.session
or "OIDC" not in request.session[BACKEND_SESSION_KEY]
or request.path in self.exempted_urls
):
return self.get_response(request)
# At that point, we know that a OIDC user was previously logged in
# and the session has expired.
# Remove authentication backend name from session to avoid being
# redirected to logout page on every subsequent GET requests
request.session.pop(BACKEND_SESSION_KEY, None)
# Redirect to a view specified in django settings.
next = request.get_full_path()
logout_url = reverse(
self.redirect_view, query_params={"next": next, "remote_user": 1}
)
return HttpResponseRedirect(logout_url)