Skip to content

Commit aeb25be

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 aef5af9 commit aeb25be

14 files changed

+202
-30
lines changed

src/onelogin/saml2/response.py

Lines changed: 62 additions & 9 deletions
Original file line numberDiff line numberDiff line change
@@ -114,31 +114,41 @@ def is_valid(self, request_data, request_id=None):
114114

115115
if security.get('wantNameIdEncrypted', False):
116116
encrypted_nameid_nodes = self.__query_assertion('/saml:Subject/saml:EncryptedID/xenc:EncryptedData')
117-
if len(encrypted_nameid_nodes) == 0:
117+
if len(encrypted_nameid_nodes) != 1:
118118
raise Exception('The NameID of the Response is not encrypted and the SP require it')
119119

120-
# Checks that there is at least one AttributeStatement if required
121-
attribute_statement_nodes = self.__query_assertion('/saml:AttributeStatement')
122-
if security.get('wantAttributeStatement', True) and not attribute_statement_nodes:
123-
raise Exception('There is no AttributeStatement on the Response')
120+
# Checks that a Conditions element exists
121+
if not self.check_one_condition():
122+
raise Exception('The Assertion must include a Conditions element')
124123

125124
# Validates Assertion timestamps
126125
if not self.validate_timestamps():
127126
raise Exception('Timing issues (please check your clock settings)')
128127

128+
# Checks that an AuthnStatement element exists and is unique
129+
if not self.check_one_authnstatement():
130+
raise Exception('The Assertion must include an AuthnStatement element')
131+
132+
# Checks that there is at least one AttributeStatement if required
133+
attribute_statement_nodes = self.__query_assertion('/saml:AttributeStatement')
134+
if security.get('wantAttributeStatement', True) and not attribute_statement_nodes:
135+
raise Exception('There is no AttributeStatement on the Response')
136+
129137
encrypted_attributes_nodes = self.__query_assertion('/saml:AttributeStatement/saml:EncryptedAttribute')
130138
if encrypted_attributes_nodes:
131139
raise Exception('There is an EncryptedAttribute in the Response and this SP not support them')
132140

133141
# Checks destination
134-
destination = self.document.get('Destination', '')
142+
destination = self.document.get('Destination', None)
135143
if destination:
136144
if not destination.startswith(current_url):
137145
# TODO: Review if following lines are required, since we can control the
138146
# request_data
139147
# current_url_routed = OneLogin_Saml2_Utils.get_self_routed_url_no_query(request_data)
140148
# if not destination.startswith(current_url_routed):
141149
raise Exception('The response was received at %s instead of %s' % (current_url, destination))
150+
elif destination == '':
151+
raise Exception('The response has an empty Destination value')
142152

143153
# Checks audience
144154
valid_audiences = self.get_audiences()
@@ -242,6 +252,26 @@ def check_status(self):
242252
status_exception_msg += ' -> ' + status_msg
243253
raise Exception(status_exception_msg)
244254

255+
def check_one_condition(self):
256+
"""
257+
Checks that the samlp:Response/saml:Assertion/saml:Conditions element exists and is unique.
258+
"""
259+
condition_nodes = self.__query_assertion('/saml:Conditions')
260+
if len(condition_nodes) == 1:
261+
return True
262+
else:
263+
return False
264+
265+
def check_one_authnstatement(self):
266+
"""
267+
Checks that the samlp:Response/saml:Assertion/saml:AuthnStatement element exists and is unique.
268+
"""
269+
authnstatement_nodes = self.__query_assertion('/saml:AuthnStatement')
270+
if len(authnstatement_nodes) == 1:
271+
return True
272+
else:
273+
return False
274+
245275
def get_audiences(self):
246276
"""
247277
Gets the audiences
@@ -262,12 +292,16 @@ def get_issuers(self):
262292
issuers = []
263293

264294
message_issuer_nodes = self.__query('/samlp:Response/saml:Issuer')
265-
if message_issuer_nodes:
295+
if len(message_issuer_nodes) == 1:
266296
issuers.append(message_issuer_nodes[0].text)
297+
else:
298+
raise Exception('Issuer of the Response not found or multiple.')
267299

268300
assertion_issuer_nodes = self.__query_assertion('/saml:Issuer')
269-
if assertion_issuer_nodes:
301+
if len(assertion_issuer_nodes) == 1:
270302
issuers.append(assertion_issuer_nodes[0].text)
303+
else:
304+
raise Exception('Issuer of the Assertion not found or multiple.')
271305

272306
return list(set(issuers))
273307

@@ -296,10 +330,19 @@ def get_nameid_data(self):
296330
if security.get('wantNameId', True):
297331
raise Exception('Not NameID found in the assertion of the Response')
298332
else:
333+
if self.__settings.is_strict() and not nameid.text:
334+
raise Exception('An empty NameID value found')
335+
299336
nameid_data = {'Value': nameid.text}
300337
for attr in ['Format', 'SPNameQualifier', 'NameQualifier']:
301338
value = nameid.get(attr, None)
302339
if value:
340+
if self.__settings.is_strict() and attr == 'SPNameQualifier':
341+
sp_data = self.__settings.get_sp_data()
342+
sp_entity_id = sp_data.get('entityId', '')
343+
if sp_entity_id != value:
344+
raise Exception('The SPNameQualifier value mistmatch the SP entityID value.')
345+
303346
nameid_data[attr] = value
304347
return nameid_data
305348

@@ -355,6 +398,9 @@ def get_attributes(self):
355398
attribute_nodes = self.__query_assertion('/saml:AttributeStatement/saml:Attribute')
356399
for attribute_node in attribute_nodes:
357400
attr_name = attribute_node.get('Name')
401+
if attr_name in attributes.keys():
402+
raise Exception('Found an Attribute element with duplicated Name')
403+
358404
values = []
359405
for attr in attribute_node.iterchildren('{%s}AttributeValue' % OneLogin_Saml2_Constants.NSMAP['saml']):
360406
# Remove any whitespace (which may be present where attributes are
@@ -386,7 +432,14 @@ def validate_num_assertions(self):
386432
"""
387433
encrypted_assertion_nodes = OneLogin_Saml2_Utils.query(self.document, '//saml:EncryptedAssertion')
388434
assertion_nodes = OneLogin_Saml2_Utils.query(self.document, '//saml:Assertion')
389-
return (len(encrypted_assertion_nodes) + len(assertion_nodes)) == 1
435+
436+
valid = len(encrypted_assertion_nodes) + len(assertion_nodes) == 1
437+
438+
if (self.encrypted):
439+
assertion_nodes = OneLogin_Saml2_Utils.query(self.decrypted_document, '//saml:Assertion')
440+
valid = valid and len(assertion_nodes) == 1
441+
442+
return valid
390443

391444
def process_signed_elements(self):
392445
"""

src/onelogin/saml2/utils.py

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

696696
status_entry = OneLogin_Saml2_Utils.query(dom, '/samlp:Response/samlp:Status')
697-
if len(status_entry) == 0:
698-
raise Exception('Missing Status on response')
697+
if len(status_entry) != 1:
698+
raise Exception('Missing valid Status on response')
699699

700700
code_entry = OneLogin_Saml2_Utils.query(dom, '/samlp:Response/samlp:Status/samlp:StatusCode', status_entry[0])
701-
if len(code_entry) == 0:
702-
raise Exception('Missing Status Code on response')
701+
if len(code_entry) != 1:
702+
raise Exception('Missing valid Status Code on response')
703703
code = code_entry[0].values()[0]
704704
status['code'] = code
705705

706+
status['msg'] = ''
706707
message_entry = OneLogin_Saml2_Utils.query(dom, '/samlp:Response/samlp:Status/samlp:StatusMessage', status_entry[0])
707708
if len(message_entry) == 0:
708709
subcode_entry = OneLogin_Saml2_Utils.query(dom, '/samlp:Response/samlp:Status/samlp:StatusCode/samlp:StatusCode', status_entry[0])
709-
if len(subcode_entry) > 0:
710+
if len(subcode_entry) == 1:
710711
status['msg'] = subcode_entry[0].values()[0]
711-
else:
712-
status['msg'] = ''
713-
else:
712+
elif len(message_entry) == 1:
714713
status['msg'] = message_entry[0].text
715714

716715
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)