2020-07-11 00:02:55 +01:00
|
|
|
"""SAML AuthnRequest Processor"""
|
2020-07-12 16:55:09 +01:00
|
|
|
from base64 import b64encode
|
2020-07-11 18:57:27 +01:00
|
|
|
from typing import Dict
|
2020-07-12 15:17:35 +01:00
|
|
|
from urllib.parse import quote_plus
|
2020-07-11 18:57:27 +01:00
|
|
|
|
2020-07-12 15:17:35 +01:00
|
|
|
from cryptography.hazmat.primitives import hashes
|
|
|
|
from cryptography.hazmat.primitives.asymmetric import padding
|
2020-07-11 00:02:55 +01:00
|
|
|
from django.http import HttpRequest
|
2020-07-11 18:57:27 +01:00
|
|
|
from lxml import etree # nosec
|
2020-07-11 00:02:55 +01:00
|
|
|
from lxml.etree import Element # nosec
|
2020-07-12 15:17:35 +01:00
|
|
|
from signxml import XMLSigner
|
2020-07-11 00:02:55 +01:00
|
|
|
|
|
|
|
from passbook.providers.saml.utils import get_random_id
|
2020-07-12 16:55:09 +01:00
|
|
|
from passbook.providers.saml.utils.encoding import deflate_and_base64_encode
|
2020-07-11 00:02:55 +01:00
|
|
|
from passbook.providers.saml.utils.time import get_time_string
|
|
|
|
from passbook.sources.saml.models import SAMLSource
|
|
|
|
from passbook.sources.saml.processors.constants import (
|
|
|
|
NS_MAP,
|
|
|
|
NS_SAML_ASSERTION,
|
|
|
|
NS_SAML_PROTOCOL,
|
|
|
|
)
|
|
|
|
|
2020-09-11 23:53:38 +01:00
|
|
|
SESSION_REQUEST_ID = "passbook_source_saml_request_id"
|
|
|
|
|
2020-07-11 00:02:55 +01:00
|
|
|
|
|
|
|
class RequestProcessor:
|
|
|
|
"""SAML AuthnRequest Processor"""
|
|
|
|
|
|
|
|
source: SAMLSource
|
|
|
|
http_request: HttpRequest
|
|
|
|
|
2020-07-12 00:46:46 +01:00
|
|
|
relay_state: str
|
|
|
|
|
2020-07-11 18:57:27 +01:00
|
|
|
request_id: str
|
|
|
|
issue_instant: str
|
|
|
|
|
2020-07-12 00:46:46 +01:00
|
|
|
def __init__(self, source: SAMLSource, request: HttpRequest, relay_state: str):
|
2020-07-11 00:02:55 +01:00
|
|
|
self.source = source
|
|
|
|
self.http_request = request
|
2020-07-12 00:46:46 +01:00
|
|
|
self.relay_state = relay_state
|
2020-07-11 18:57:27 +01:00
|
|
|
self.request_id = get_random_id()
|
2020-09-11 23:53:38 +01:00
|
|
|
self.http_request.session[SESSION_REQUEST_ID] = self.request_id
|
2020-07-11 18:57:27 +01:00
|
|
|
self.issue_instant = get_time_string()
|
2020-07-11 00:02:55 +01:00
|
|
|
|
|
|
|
def get_issuer(self) -> Element:
|
|
|
|
"""Get Issuer Element"""
|
|
|
|
issuer = Element(f"{{{NS_SAML_ASSERTION}}}Issuer")
|
|
|
|
issuer.text = self.source.get_issuer(self.http_request)
|
|
|
|
return issuer
|
|
|
|
|
|
|
|
def get_name_id_policy(self) -> Element:
|
|
|
|
"""Get NameID Policy Element"""
|
|
|
|
name_id_policy = Element(f"{{{NS_SAML_PROTOCOL}}}NameIDPolicy")
|
|
|
|
name_id_policy.text = self.source.name_id_policy
|
|
|
|
return name_id_policy
|
|
|
|
|
2020-07-11 18:57:27 +01:00
|
|
|
def get_auth_n(self) -> Element:
|
2020-07-11 00:02:55 +01:00
|
|
|
"""Get full AuthnRequest"""
|
|
|
|
auth_n_request = Element(f"{{{NS_SAML_PROTOCOL}}}AuthnRequest", nsmap=NS_MAP)
|
|
|
|
auth_n_request.attrib[
|
|
|
|
"AssertionConsumerServiceURL"
|
|
|
|
] = self.source.build_full_url(self.http_request)
|
|
|
|
auth_n_request.attrib["Destination"] = self.source.sso_url
|
2020-07-11 18:57:27 +01:00
|
|
|
auth_n_request.attrib["ID"] = self.request_id
|
|
|
|
auth_n_request.attrib["IssueInstant"] = self.issue_instant
|
2020-07-11 00:02:55 +01:00
|
|
|
auth_n_request.attrib["ProtocolBinding"] = self.source.binding_type
|
|
|
|
auth_n_request.attrib["Version"] = "2.0"
|
|
|
|
# Create issuer object
|
|
|
|
auth_n_request.append(self.get_issuer())
|
|
|
|
# Create NameID Policy Object
|
|
|
|
auth_n_request.append(self.get_name_id_policy())
|
2020-07-11 18:57:27 +01:00
|
|
|
return auth_n_request
|
|
|
|
|
|
|
|
def build_auth_n(self) -> str:
|
|
|
|
"""Get Signed string representation of AuthN Request
|
|
|
|
(used for POST Bindings)"""
|
|
|
|
auth_n_request = self.get_auth_n()
|
|
|
|
|
|
|
|
if self.source.signing_kp:
|
|
|
|
signed_request = XMLSigner().sign(
|
|
|
|
auth_n_request,
|
|
|
|
cert=self.source.signing_kp.certificate_data,
|
|
|
|
key=self.source.signing_kp.key_data,
|
|
|
|
)
|
|
|
|
return etree.tostring(signed_request).decode()
|
|
|
|
|
|
|
|
return etree.tostring(auth_n_request).decode()
|
|
|
|
|
|
|
|
def build_auth_n_detached(self) -> Dict[str, str]:
|
|
|
|
"""Get Dict AuthN Request for Redirect bindings, with detached
|
2020-07-12 15:17:35 +01:00
|
|
|
Signature. See https://docs.oasis-open.org/security/saml/v2.0/saml-bindings-2.0-os.pdf"""
|
2020-07-11 18:57:27 +01:00
|
|
|
auth_n_request = self.get_auth_n()
|
|
|
|
|
2020-07-12 15:17:35 +01:00
|
|
|
saml_request = deflate_and_base64_encode(
|
|
|
|
etree.tostring(auth_n_request).decode()
|
|
|
|
)
|
|
|
|
|
2020-07-11 18:57:27 +01:00
|
|
|
response_dict = {
|
2020-07-12 15:17:35 +01:00
|
|
|
"SAMLRequest": saml_request,
|
2020-07-11 18:57:27 +01:00
|
|
|
}
|
|
|
|
|
2020-07-12 15:17:35 +01:00
|
|
|
if self.relay_state != "":
|
|
|
|
response_dict["RelayState"] = self.relay_state
|
|
|
|
|
2020-07-11 18:57:27 +01:00
|
|
|
if self.source.signing_kp:
|
2020-07-12 15:17:35 +01:00
|
|
|
sig_alg = "http://www.w3.org/2000/09/xmldsig#rsa-sha1"
|
|
|
|
sig_hash = hashes.SHA1() # nosec
|
|
|
|
# Create the full querystring in the correct order to be signed
|
|
|
|
querystring = f"SAMLRequest={quote_plus(saml_request)}&"
|
|
|
|
if self.relay_state != "":
|
|
|
|
querystring += f"RelayState={quote_plus(self.relay_state)}&"
|
|
|
|
querystring += f"SigAlg={sig_alg}"
|
|
|
|
|
|
|
|
signature = self.source.signing_kp.private_key.sign(
|
|
|
|
querystring.encode(),
|
|
|
|
padding.PSS(
|
|
|
|
mgf=padding.MGF1(sig_hash), salt_length=padding.PSS.MAX_LENGTH
|
|
|
|
),
|
|
|
|
sig_hash,
|
2020-07-11 18:57:27 +01:00
|
|
|
)
|
2020-07-12 15:17:35 +01:00
|
|
|
response_dict["SigAlg"] = sig_alg
|
2020-07-12 16:55:09 +01:00
|
|
|
response_dict["Signature"] = b64encode(signature).decode()
|
2020-07-11 18:57:27 +01:00
|
|
|
|
|
|
|
return response_dict
|