This repository has been archived on 2024-05-31. You can view files and clone it, but cannot push or open issues or pull requests.
authentik/passbook/channels/in_saml/views.py

113 lines
4.2 KiB
Python
Raw Normal View History

2019-11-07 16:02:56 +00:00
"""saml sp views"""
from django.contrib.auth import login, logout
2019-11-07 17:02:59 +00:00
from django.http import Http404, HttpRequest, HttpResponse
2019-11-07 16:02:56 +00:00
from django.shortcuts import get_object_or_404, redirect, render, reverse
from django.utils.decorators import method_decorator
from django.views import View
from django.views.decorators.csrf import csrf_exempt
from signxml.util import strip_pem_header
2019-11-07 16:02:56 +00:00
from passbook.channels.in_saml.exceptions import (
MissingSAMLResponse,
UnsupportedNameIDFormat,
2019-12-31 11:51:16 +00:00
)
from passbook.channels.in_saml.models import SAMLInlet
from passbook.channels.in_saml.processors.base import Processor
from passbook.channels.in_saml.utils import build_full_url, get_issuer
from passbook.channels.in_saml.xml_render import get_authnrequest_xml
from passbook.channels.out_saml.utils import get_random_id, render_xml
from passbook.channels.out_saml.utils.encoding import nice64
from passbook.channels.out_saml.utils.time import get_time_string
from passbook.lib.views import bad_request_message
2019-11-07 16:02:56 +00:00
class InitiateView(View):
"""Get the Form with SAML Request, which sends us to the IDP"""
def get(self, request: HttpRequest, inlet_slug: str) -> HttpResponse:
2019-11-07 16:02:56 +00:00
"""Replies with an XHTML SSO Request."""
inlet: SAMLInlet = get_object_or_404(SAMLInlet, slug=inlet_slug)
if not inlet.enabled:
raise Http404
2019-12-31 11:51:16 +00:00
sso_destination = request.GET.get("next", None)
request.session["sso_destination"] = sso_destination
2019-11-07 16:02:56 +00:00
parameters = {
"ACS_URL": build_full_url("acs", request, inlet),
"DESTINATION": inlet.idp_url,
2019-12-31 11:51:16 +00:00
"AUTHN_REQUEST_ID": get_random_id(),
"ISSUE_INSTANT": get_time_string(),
"ISSUER": get_issuer(request, inlet),
2019-11-07 16:02:56 +00:00
}
authn_req = get_authnrequest_xml(parameters, signed=False)
_request = nice64(str.encode(authn_req))
2019-12-31 11:51:16 +00:00
return render(
request,
"saml/sp/login.html",
{
"request_url": inlet.idp_url,
2019-12-31 11:51:16 +00:00
"request": _request,
"token": sso_destination,
"inlet": inlet,
2019-12-31 11:51:16 +00:00
},
)
2019-11-07 16:02:56 +00:00
2019-12-31 11:51:16 +00:00
@method_decorator(csrf_exempt, name="dispatch")
2019-11-07 16:02:56 +00:00
class ACSView(View):
"""AssertionConsumerService, consume assertion and log user in"""
def post(self, request: HttpRequest, inlet_slug: str) -> HttpResponse:
2019-11-07 16:02:56 +00:00
"""Handles a POSTed SSO Assertion and logs the user in."""
inlet: SAMLInlet = get_object_or_404(SAMLInlet, slug=inlet_slug)
if not inlet.enabled:
raise Http404
processor = Processor(inlet)
try:
processor.parse(request)
except MissingSAMLResponse as exc:
return bad_request_message(request, str(exc))
try:
user = processor.get_user()
login(request, user, backend="django.contrib.auth.backends.ModelBackend")
return redirect(reverse("passbook_core:overview"))
except UnsupportedNameIDFormat as exc:
return bad_request_message(request, str(exc))
2019-11-07 16:02:56 +00:00
class SLOView(View):
"""Single-Logout-View"""
def dispatch(self, request: HttpRequest, inlet_slug: str) -> HttpResponse:
2019-11-07 16:02:56 +00:00
"""Replies with an XHTML SSO Request."""
inlet: SAMLInlet = get_object_or_404(SAMLInlet, slug=inlet_slug)
if not inlet.enabled:
raise Http404
2019-11-07 16:02:56 +00:00
logout(request)
2019-12-31 11:51:16 +00:00
return render(
request,
"saml/sp/sso_single_logout.html",
{"idp_logout_url": inlet.idp_logout_url, "autosubmit": inlet.auto_logout,},
2019-12-31 11:51:16 +00:00
)
2019-11-07 16:02:56 +00:00
class MetadataView(View):
"""Return XML Metadata for IDP"""
def dispatch(self, request: HttpRequest, inlet_slug: str) -> HttpResponse:
2019-11-07 16:02:56 +00:00
"""Replies with the XML Metadata SPSSODescriptor."""
inlet: SAMLInlet = get_object_or_404(SAMLInlet, slug=inlet_slug)
issuer = get_issuer(request, inlet)
2020-03-03 22:35:38 +00:00
cert_stripped = strip_pem_header(
inlet.signing_kp.certificate_data.replace("\r", "")
2020-03-03 22:35:38 +00:00
).replace("\n", "")
2019-12-31 11:51:16 +00:00
return render_xml(
request,
2020-02-20 16:23:27 +00:00
"saml/sp/xml/sp_sso_descriptor.xml",
2019-12-31 11:51:16 +00:00
{
"acs_url": build_full_url("acs", request, inlet),
2020-02-20 16:23:27 +00:00
"issuer": issuer,
"cert_public_key": cert_stripped,
2019-12-31 11:51:16 +00:00
},
)