Skip to content

Commit e239bd5

Browse files
committed
Several security improvements:
- Conditions element required and unique. - AuthnStatement element required and unique. - SPNameQualifier must math the SP EntityID - Reject saml:Attribute element with same “Name” attribute - Reject empty nameID - Require Issuer element. (Must match IdP EntityID). - Destination value can't be blank (if present must match ACS URL). - Check that the EncryptedAssertion element only contains 1 Assertion element.
1 parent 45a373d commit e239bd5

14 files changed

Lines changed: 221 additions & 32 deletions

src/onelogin/saml2/response.py

Lines changed: 63 additions & 10 deletions
Original file line numberDiff line numberDiff line change
@@ -110,31 +110,41 @@ def is_valid(self, request_data, request_id=None):
110110

111111
if security['wantNameIdEncrypted']:
112112
encrypted_nameid_nodes = self.__query_assertion('/saml:Subject/saml:EncryptedID/xenc:EncryptedData')
113-
if len(encrypted_nameid_nodes) == 0:
113+
if len(encrypted_nameid_nodes) != 1:
114114
raise Exception('The NameID of the Response is not encrypted and the SP require it')
115115

116-
# Checks that there is at least one AttributeStatement if required
117-
attribute_statement_nodes = self.__query_assertion('/saml:AttributeStatement')
118-
if security['wantAttributeStatement'] and not attribute_statement_nodes:
119-
raise Exception('There is no AttributeStatement on the Response')
116+
# Checks that a Conditions element exists
117+
if not self.check_one_condition():
118+
raise Exception('The Assertion must include a Conditions element')
120119

121120
# Validates Assertion timestamps
122121
if not self.validate_timestamps():
123122
raise Exception('Timing issues (please check your clock settings)')
124123

124+
# Checks that an AuthnStatement element exists and is unique
125+
if not self.check_one_authnstatement():
126+
raise Exception('The Assertion must include an AuthnStatement element')
127+
128+
# Checks that there is at least one AttributeStatement if required
129+
attribute_statement_nodes = self.__query_assertion('/saml:AttributeStatement')
130+
if security.get('wantAttributeStatement', True) and not attribute_statement_nodes:
131+
raise Exception('There is no AttributeStatement on the Response')
132+
125133
encrypted_attributes_nodes = self.__query_assertion('/saml:AttributeStatement/saml:EncryptedAttribute')
126134
if encrypted_attributes_nodes:
127135
raise Exception('There is an EncryptedAttribute in the Response and this SP not support them')
128136

129137
# Checks destination
130-
destination = self.document.get('Destination', '')
138+
destination = self.document.get('Destination', None)
131139
if destination:
132140
if not destination.startswith(current_url):
133141
# TODO: Review if following lines are required, since we can control the
134142
# request_data
135143
# current_url_routed = OneLogin_Saml2_Utils.get_self_routed_url_no_query(request_data)
136144
# if not destination.startswith(current_url_routed):
137145
raise Exception('The response was received at %s instead of %s' % (current_url, destination))
146+
elif destination == '':
147+
raise Exception('The response has an empty Destination value')
138148

139149
# Checks audience
140150
valid_audiences = self.get_audiences()
@@ -239,6 +249,26 @@ def check_status(self):
239249
status_exception_msg += ' -> ' + status_msg
240250
raise Exception(status_exception_msg)
241251

252+
def check_one_condition(self):
253+
"""
254+
Checks that the samlp:Response/saml:Assertion/saml:Conditions element exists and is unique.
255+
"""
256+
condition_nodes = self.__query_assertion('/saml:Conditions')
257+
if len(condition_nodes) == 1:
258+
return True
259+
else:
260+
return False
261+
262+
def check_one_authnstatement(self):
263+
"""
264+
Checks that the samlp:Response/saml:Assertion/saml:AuthnStatement element exists and is unique.
265+
"""
266+
authnstatement_nodes = self.__query_assertion('/saml:AuthnStatement')
267+
if len(authnstatement_nodes) == 1:
268+
return True
269+
else:
270+
return False
271+
242272
def get_audiences(self):
243273
"""
244274
Gets the audiences
@@ -259,14 +289,18 @@ def get_issuers(self):
259289
issuers = set()
260290

261291
message_issuer_nodes = self.__query('/samlp:Response/saml:Issuer')
262-
if message_issuer_nodes:
292+
if len(message_issuer_nodes) == 1:
263293
issuers.add(message_issuer_nodes[0].text)
294+
else:
295+
raise Exception('Issuer of the Response not found or multiple.')
264296

265297
assertion_issuer_nodes = self.__query_assertion('/saml:Issuer')
266-
if assertion_issuer_nodes:
298+
if len(assertion_issuer_nodes) == 1:
267299
issuers.add(assertion_issuer_nodes[0].text)
300+
else:
301+
raise Exception('Issuer of the Assertion not found or multiple.')
268302

269-
return list(issuers)
303+
return list(set(issuers))
270304

271305
def get_nameid_data(self):
272306
"""
@@ -292,10 +326,19 @@ def get_nameid_data(self):
292326
if security.get('wantNameId', True):
293327
raise Exception('Not NameID found in the assertion of the Response')
294328
else:
329+
if self.__settings.is_strict() and not nameid.text:
330+
raise Exception('An empty NameID value found')
331+
295332
nameid_data = {'Value': nameid.text}
296333
for attr in ['Format', 'SPNameQualifier', 'NameQualifier']:
297334
value = nameid.get(attr, None)
298335
if value:
336+
if self.__settings.is_strict() and attr == 'SPNameQualifier':
337+
sp_data = self.__settings.get_sp_data()
338+
sp_entity_id = sp_data.get('entityId', '')
339+
if sp_entity_id != value:
340+
raise Exception('The SPNameQualifier value mistmatch the SP entityID value.')
341+
299342
nameid_data[attr] = value
300343
return nameid_data
301344

@@ -351,6 +394,9 @@ def get_attributes(self):
351394
attribute_nodes = self.__query_assertion('/saml:AttributeStatement/saml:Attribute')
352395
for attribute_node in attribute_nodes:
353396
attr_name = attribute_node.get('Name')
397+
if attr_name in attributes.keys():
398+
raise Exception('Found an Attribute element with duplicated Name')
399+
354400
values = []
355401
for attr in attribute_node.iterchildren('{%s}AttributeValue' % OneLogin_Saml2_Constants.NSMAP['saml']):
356402
values.append(attr.text)
@@ -366,7 +412,14 @@ def validate_num_assertions(self):
366412
"""
367413
encrypted_assertion_nodes = OneLogin_Saml2_XML.query(self.document, '//saml:EncryptedAssertion')
368414
assertion_nodes = OneLogin_Saml2_XML.query(self.document, '//saml:Assertion')
369-
return (len(encrypted_assertion_nodes) + len(assertion_nodes)) == 1
415+
416+
valid = len(encrypted_assertion_nodes) + len(assertion_nodes) == 1
417+
418+
if (self.encrypted):
419+
assertion_nodes = OneLogin_Saml2_XML.query(self.decrypted_document, '//saml:Assertion')
420+
valid = valid and len(assertion_nodes) == 1
421+
422+
return valid
370423

371424
def process_signed_elements(self):
372425
"""

src/onelogin/saml2/utils.py

Lines changed: 7 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -604,23 +604,22 @@ def get_status(dom):
604604
status = {}
605605

606606
status_entry = OneLogin_Saml2_XML.query(dom, '/samlp:Response/samlp:Status')
607-
if len(status_entry) == 0:
608-
raise Exception('Missing Status on response')
607+
if len(status_entry) != 1:
608+
raise Exception('Missing valid Status on response')
609609

610610
code_entry = OneLogin_Saml2_XML.query(dom, '/samlp:Response/samlp:Status/samlp:StatusCode', status_entry[0])
611-
if len(code_entry) == 0:
612-
raise Exception('Missing Status Code on response')
611+
if len(code_entry) != 1:
612+
raise Exception('Missing valid Status Code on response')
613613
code = code_entry[0].values()[0]
614614
status['code'] = code
615615

616+
status['msg'] = ''
616617
message_entry = OneLogin_Saml2_XML.query(dom, '/samlp:Response/samlp:Status/samlp:StatusMessage', status_entry[0])
617618
if len(message_entry) == 0:
618619
subcode_entry = OneLogin_Saml2_XML.query(dom, '/samlp:Response/samlp:Status/samlp:StatusCode/samlp:StatusCode', status_entry[0])
619-
if len(subcode_entry) > 0:
620+
if len(subcode_entry) == 1:
620621
status['msg'] = subcode_entry[0].values()[0]
621-
else:
622-
status['msg'] = ''
623-
else:
622+
elif len(message_entry) == 1:
624623
status['msg'] = message_entry[0].text
625624

626625
return status
Lines changed: 1 addition & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
<?xml version="1.0"?>
<samlp:Response xmlns:samlp="urn:oasis:names:tc:SAML:2.0:protocol" xmlns:saml="urn:oasis:names:tc:SAML:2.0:assertion" ID="pfx44992ebb-4b38-e432-db82-9952410d9aab" Version="2.0" IssueInstant="2014-03-21T13:42:31Z" Destination="https://pitbulk.no-ip.org/newonelogin/demo1/index.php?acs" InResponseTo="ONELOGIN_191c03e68d71d9796f5e07e6262ca4ad883a74b1"><saml:Issuer>https://pitbulk.no-ip.org/simplesaml/saml2/idp/metadata.php</saml:Issuer><ds:Signature xmlns:ds="http://www.w3.org/2000/09/xmldsig#">
  <ds:SignedInfo><ds:CanonicalizationMethod Algorithm="http://www.w3.org/2001/10/xml-exc-c14n#"/>
    <ds:SignatureMethod Algorithm="http://www.w3.org/2000/09/xmldsig#rsa-sha1"/>
  <ds:Reference URI="#pfx44992ebb-4b38-e432-db82-9952410d9aab"><ds:Transforms><ds:Transform Algorithm="http://www.w3.org/2000/09/xmldsig#enveloped-signature"/><ds:Transform Algorithm="http://www.w3.org/2001/10/xml-exc-c14n#"/></ds:Transforms><ds:DigestMethod Algorithm="http://www.w3.org/2000/09/xmldsig#sha1"/><ds:DigestValue>gvRrrgxpAdylIA/2srFmJd+jis8=</ds:DigestValue></ds:Reference></ds:SignedInfo><ds:SignatureValue>Kdp8T8rnwPcBUohcqPM0eiNXpMh3lc+epHTDHqLEnOJrgu5/jj+i7EaAmgO0RJTkhDEY0V8FneT4vovcAbg9fbM8fTO1lX82wImsEdq2L3SE84qBuaCmDV5Yo07CHbQOQjaetTktJuoF08Ad6l+5hRO/pJxmrEyG+4KihFYBuuk=</ds:SignatureValue>
<ds:KeyInfo><ds:X509Data><ds:X509Certificate>MIICgTCCAeoCCQCbOlrWDdX7FTANBgkqhkiG9w0BAQUFADCBhDELMAkGA1UEBhMCTk8xGDAWBgNVBAgTD0FuZHJlYXMgU29sYmVyZzEMMAoGA1UEBxMDRm9vMRAwDgYDVQQKEwdVTklORVRUMRgwFgYDVQQDEw9mZWlkZS5lcmxhbmcubm8xITAfBgkqhkiG9w0BCQEWEmFuZHJlYXNAdW5pbmV0dC5ubzAeFw0wNzA2MTUxMjAxMzVaFw0wNzA4MTQxMjAxMzVaMIGEMQswCQYDVQQGEwJOTzEYMBYGA1UECBMPQW5kcmVhcyBTb2xiZXJnMQwwCgYDVQQHEwNGb28xEDAOBgNVBAoTB1VOSU5FVFQxGDAWBgNVBAMTD2ZlaWRlLmVybGFuZy5ubzEhMB8GCSqGSIb3DQEJARYSYW5kcmVhc0B1bmluZXR0Lm5vMIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDivbhR7P516x/S3BqKxupQe0LONoliupiBOesCO3SHbDrl3+q9IbfnfmE04rNuMcPsIxB161TdDpIesLCn7c8aPHISKOtPlAeTZSnb8QAu7aRjZq3+PbrP5uW3TcfCGPtKTytHOge/OlJbo078dVhXQ14d1EDwXJW1rRXuUt4C8QIDAQABMA0GCSqGSIb3DQEBBQUAA4GBACDVfp86HObqY+e8BUoWQ9+VMQx1ASDohBjwOsg2WykUqRXF+dLfcUH9dWR63CtZIKFDbStNomPnQz7nbK+onygwBspVEbnHuUihZq3ZUdmumQqCw4Uvs/1Uvq3orOo/WJVhTyvLgFVK2QarQ4/67OZfHd7R+POBXhophSMv1ZOo</ds:X509Certificate></ds:X509Data></ds:KeyInfo></ds:Signature><samlp:Status><samlp:StatusCode Value="urn:oasis:names:tc:SAML:2.0:status:Success"/></samlp:Status><saml:Assertion xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:xs="http://www.w3.org/2001/XMLSchema" ID="pfx80baaef6-292b-8747-cfca-de1ee3f1a415" Version="2.0" IssueInstant="2014-03-21T13:42:31Z"><saml:Issuer>https://pitbulk.no-ip.org/simplesaml/saml2/idp/metadata.php</saml:Issuer><ds:Signature xmlns:ds="http://www.w3.org/2000/09/xmldsig#">
  <ds:SignedInfo><ds:CanonicalizationMethod Algorithm="http://www.w3.org/2001/10/xml-exc-c14n#"/>
    <ds:SignatureMethod Algorithm="http://www.w3.org/2000/09/xmldsig#rsa-sha1"/>
  <ds:Reference URI="#pfx80baaef6-292b-8747-cfca-de1ee3f1a415"><ds:Transforms><ds:Transform Algorithm="http://www.w3.org/2000/09/xmldsig#enveloped-signature"/><ds:Transform Algorithm="http://www.w3.org/2001/10/xml-exc-c14n#"/></ds:Transforms><ds:DigestMethod Algorithm="http://www.w3.org/2000/09/xmldsig#sha1"/><ds:DigestValue>aR9M4ewNs3u+nJaQCD26Z0AwD6M=</ds:DigestValue></ds:Reference></ds:SignedInfo><ds:SignatureValue>4d8XJ5mpNimoBHdzsWf/ZzlUNQ7JiUxIx+PyN4n3A/ma1pl/CAOIKNS6trTzI897VcllgxXaM9cPVj9HKaOZEn0HNPkaVGucyUOW1TwgVvrUvCMAuQO7QgmZzGuIXlnUJKqiL4Y18MOS5TjKhLhHn1la8LAnrdUTBhmLyxkcf8U=</ds:SignatureValue>
<ds:KeyInfo><ds:X509Data><ds:X509Certificate>MIICgTCCAeoCCQCbOlrWDdX7FTANBgkqhkiG9w0BAQUFADCBhDELMAkGA1UEBhMCTk8xGDAWBgNVBAgTD0FuZHJlYXMgU29sYmVyZzEMMAoGA1UEBxMDRm9vMRAwDgYDVQQKEwdVTklORVRUMRgwFgYDVQQDEw9mZWlkZS5lcmxhbmcubm8xITAfBgkqhkiG9w0BCQEWEmFuZHJlYXNAdW5pbmV0dC5ubzAeFw0wNzA2MTUxMjAxMzVaFw0wNzA4MTQxMjAxMzVaMIGEMQswCQYDVQQGEwJOTzEYMBYGA1UECBMPQW5kcmVhcyBTb2xiZXJnMQwwCgYDVQQHEwNGb28xEDAOBgNVBAoTB1VOSU5FVFQxGDAWBgNVBAMTD2ZlaWRlLmVybGFuZy5ubzEhMB8GCSqGSIb3DQEJARYSYW5kcmVhc0B1bmluZXR0Lm5vMIGfMA0GCSqGSIb3DQEBAQUAA4GNADCBiQKBgQDivbhR7P516x/S3BqKxupQe0LONoliupiBOesCO3SHbDrl3+q9IbfnfmE04rNuMcPsIxB161TdDpIesLCn7c8aPHISKOtPlAeTZSnb8QAu7aRjZq3+PbrP5uW3TcfCGPtKTytHOge/OlJbo078dVhXQ14d1EDwXJW1rRXuUt4C8QIDAQABMA0GCSqGSIb3DQEBBQUAA4GBACDVfp86HObqY+e8BUoWQ9+VMQx1ASDohBjwOsg2WykUqRXF+dLfcUH9dWR63CtZIKFDbStNomPnQz7nbK+onygwBspVEbnHuUihZq3ZUdmumQqCw4Uvs/1Uvq3orOo/WJVhTyvLgFVK2QarQ4/67OZfHd7R+POBXhophSMv1ZOo</ds:X509Certificate></ds:X509Data></ds:KeyInfo></ds:Signature><saml:Subject><saml:NameID SPNameQualifier="https://pitbulk.no-ip.org/newonelogin/demo1/metadata.php" Format="urn:oasis:names:tc:SAML:2.0:nameid-format:transient">_2126dd19b8a9a28238d88fdc7385e60995004a7782</saml:NameID><saml:SubjectConfirmation Method="urn:oasis:names:tc:SAML:2.0:cm:bearer"><saml:SubjectConfirmationData NotOnOrAfter="2023-09-22T19:02:31Z" Recipient="https://pitbulk.no-ip.org/newonelogin/demo1/index.php?acs" InResponseTo="ONELOGIN_191c03e68d71d9796f5e07e6262ca4ad883a74b1"/></saml:SubjectConfirmation></saml:Subject><saml:Conditions NotBefore="2014-03-21T13:42:01Z" NotOnOrAfter="2023-09-22T19:02:31Z"><saml:AudienceRestriction><saml:Audience>https://pitbulk.no-ip.org/newonelogin/demo1/metadata.php</saml:Audience></saml:AudienceRestriction></saml:Conditions><saml:AuthnStatement AuthnInstant="2014-03-21T13:41:09Z" SessionNotOnOrAfter="2014-03-21T21:42:31Z" SessionIndex="_e6578d6af97b9f7f0672d850d29db4add1a286dc24"><saml:AuthnContext><saml:AuthnContextClassRef>urn:oasis:names:tc:SAML:2.0:ac:classes:Password</saml:AuthnContextClassRef></saml:AuthnContext></saml:AuthnStatement><saml:AttributeStatement><saml:Attribute Name="uid" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:basic"><saml:AttributeValue xsi:type="xs:string">test</saml:AttributeValue></saml:Attribute><saml:Attribute Name="uid" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:basic"><saml:AttributeValue xsi:type="xs:string">test2</saml:AttributeValue></saml:Attribute><saml:Attribute Name="mail" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:basic"><saml:AttributeValue xsi:type="xs:string">test@example.com</saml:AttributeValue></saml:Attribute><saml:Attribute Name="cn" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:basic"><saml:AttributeValue xsi:type="xs:string">test</saml:AttributeValue></saml:Attribute><saml:Attribute Name="sn" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:basic"><saml:AttributeValue xsi:type="xs:string">waa2</saml:AttributeValue></saml:Attribute><saml:Attribute Name="eduPersonAffiliation" NameFormat="urn:oasis:names:tc:SAML:2.0:attrname-format:basic"><saml:AttributeValue xsi:type="xs:string">user</saml:AttributeValue><saml:AttributeValue xsi:type="xs:string">admin</saml:AttributeValue></saml:Attribute></saml:AttributeStatement></saml:Assertion></samlp:Response>

0 commit comments

Comments
 (0)