Skip to content

Commit 53082b5

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 e54e4e2 commit 53082b5

16 files changed

+207
-25
lines changed

lib/Saml2/LogoutResponse.php

Lines changed: 1 addition & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -77,7 +77,7 @@ public function getIssuer()
7777
public function getStatus()
7878
{
7979
$entries = $this->_query('/samlp:LogoutResponse/samlp:Status/samlp:StatusCode');
80-
if ($entries->length == 0) {
80+
if ($entries->length != 1) {
8181
return null;
8282
}
8383
$status = $entries->item(0)->getAttribute('Value');

lib/Saml2/Response.php

Lines changed: 76 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -149,17 +149,27 @@ public function isValid($requestId = null)
149149

150150
if ($security['wantNameIdEncrypted']) {
151151
$encryptedIdNodes = $this->_queryAssertion('/saml:Subject/saml:EncryptedID/xenc:EncryptedData');
152-
if ($encryptedIdNodes->length == 0) {
152+
if ($encryptedIdNodes->length != 1) {
153153
throw new Exception("The NameID of the Response is not encrypted and the SP requires it");
154154
}
155155
}
156156

157+
// Validate Conditions element exists
158+
if (!$this->checkOneCondition()) {
159+
throw new Exception("The Assertion must include a Conditions element");
160+
}
161+
157162
// Validate Asserion timestamps
158163
$validTimestamps = $this->validateTimestamps();
159164
if (!$validTimestamps) {
160165
throw new Exception('Timing issues (please check your clock settings)');
161166
}
162167

168+
// Validate AuthnStatement element exists and is unique
169+
if (!$this->checkOneAuthnStatement()) {
170+
throw new Exception("The Assertion must include an AuthnStatement element");
171+
}
172+
163173
// EncryptedAttributes are not supported
164174
$encryptedAttributeNodes = $this->_queryAssertion('/saml:AttributeStatement/saml:EncryptedAttribute');
165175
if ($encryptedAttributeNodes->length > 0) {
@@ -169,7 +179,9 @@ public function isValid($requestId = null)
169179
// Check destination
170180
if ($this->document->documentElement->hasAttribute('Destination')) {
171181
$destination = trim($this->document->documentElement->getAttribute('Destination'));
172-
if (!empty($destination)) {
182+
if (empty($destination)) {
183+
throw new Exception("The response has an empty Destination value");
184+
} else {
173185
if (strpos($destination, $currentURL) !== 0) {
174186
$currentURLNoRouted = OneLogin_Saml2_Utils::getSelfURLNoQuery();
175187

@@ -316,6 +328,34 @@ public function checkStatus()
316328
}
317329
}
318330

331+
/**
332+
* Checks that the samlp:Response/saml:Assertion/saml:Conditions element exists and is unique.
333+
*
334+
* @return boolean true if the Conditions element exists and is unique
335+
*/
336+
public function checkOneCondition() {
337+
$entries = $this->_queryAssertion("/saml:Conditions");
338+
if ($entries->length == 1) {
339+
return true;
340+
} else {
341+
return false;
342+
}
343+
}
344+
345+
/**
346+
* Checks that the samlp:Response/saml:Assertion/saml:AuthnStatement element exists and is unique.
347+
*
348+
* @return boolean true if the AuthnStatement element exists and is unique
349+
*/
350+
public function checkOneAuthnStatement() {
351+
$entries = $this->_queryAssertion("/saml:AuthnStatement");
352+
if ($entries->length == 1) {
353+
return true;
354+
} else {
355+
return false;
356+
}
357+
}
358+
319359
/**
320360
* Gets the audiences.
321361
*
@@ -348,11 +388,15 @@ public function getIssuers()
348388
$responseIssuer = $this->_query('/samlp:Response/saml:Issuer');
349389
if ($responseIssuer->length == 1) {
350390
$issuers[] = $responseIssuer->item(0)->textContent;
391+
} else {
392+
throw new Exception("Issuer of the Response not found or multiple.");
351393
}
352394

353395
$assertionIssuer = $this->_queryAssertion('/saml:Issuer');
354396
if ($assertionIssuer->length == 1) {
355397
$issuers[] = $assertionIssuer->item(0)->textContent;
398+
} else {
399+
throw new Exception("Issuer of the Assertion not found or multiple.");
356400
}
357401

358402
return array_unique($issuers);
@@ -391,9 +435,20 @@ public function getNameIdData()
391435
throw new Exception("Not NameID found in the assertion of the Response");
392436
}
393437
} else {
438+
if (empty($nameId->nodeValue)) {
439+
throw new Exception("An empty NameID value found");
440+
}
394441
$nameIdData['Value'] = $nameId->nodeValue;
442+
395443
foreach (array('Format', 'SPNameQualifier', 'NameQualifier') as $attr) {
396444
if ($nameId->hasAttribute($attr)) {
445+
if ($attr == 'SPNameQualifier') {
446+
$spData = $this->_settings->getSPData();
447+
$spEntityId = $spData['entityId'];
448+
if ($spEntityId != $nameId->getAttribute($attr)) {
449+
throw new Exception("The SPNameQualifier value mistmatch the SP entityID value.");
450+
}
451+
}
397452
$nameIdData[$attr] = $nameId->getAttribute($attr);
398453
}
399454
}
@@ -476,11 +531,18 @@ public function getAttributes()
476531
*/
477532

478533
$entries = $this->_queryAssertion('/saml:AttributeStatement/saml:Attribute');
534+
$processedAttrNames = array();
479535

480536
/** @var $entry DOMNode */
481537
foreach ($entries as $entry) {
482538
$attributeName = $entry->attributes->getNamedItem('Name')->nodeValue;
483539

540+
if (in_array($attributeName, $processedAttrNames)) {
541+
throw new Exception("Found an Attribute element with duplicated Name");
542+
} else {
543+
$processedAttrNames[] = $attributeName;
544+
}
545+
484546
$attributeValues = array();
485547
foreach ($entry->childNodes as $childNode) {
486548
$tagName = ($childNode->prefix ? $childNode->prefix.':' : '') . 'AttributeValue';
@@ -503,7 +565,15 @@ public function validateNumAssertions()
503565
{
504566
$encryptedAssertionNodes = $this->document->getElementsByTagName('EncryptedAssertion');
505567
$assertionNodes = $this->document->getElementsByTagName('Assertion');
506-
return ($assertionNodes->length + $encryptedAssertionNodes->length == 1);
568+
569+
$valid = $assertionNodes->length + $encryptedAssertionNodes->length == 1;
570+
571+
if ($this->encrypted) {
572+
$assertionNodes = $this->decryptedDocument->getElementsByTagName('Assertion');
573+
$valid = $valid && $assertionNodes->length == 1;
574+
}
575+
576+
return $valid;
507577
}
508578

509579
/**
@@ -543,7 +613,7 @@ public function processSignedElements()
543613
$verifiedIds[] = $idValue;
544614

545615
$ref = $signNode->getElementsByTagName('Reference');
546-
if ($ref->length > 0) {
616+
if ($ref->length == 1) {
547617
$ref = $ref->item(0);
548618
$sei = $ref->getAttribute('URI');
549619
if (!empty($sei)) {
@@ -558,6 +628,8 @@ public function processSignedElements()
558628
}
559629
$verifiedSeis[] = $sei;
560630
}
631+
} else {
632+
throw new Exception('Unexpected number of Reference nodes found for signature. SAML Response rejected.');
561633
}
562634
$signedElements[] = $signedElement;
563635
}

lib/Saml2/Utils.php

Lines changed: 7 additions & 8 deletions
Original file line numberDiff line numberDiff line change
@@ -840,26 +840,25 @@ public static function getStatus($dom)
840840
$status = array();
841841

842842
$statusEntry = self::query($dom, '/samlp:Response/samlp:Status');
843-
if ($statusEntry->length == 0) {
844-
throw new Exception('Missing Status on response');
843+
if ($statusEntry->length != 1) {
844+
throw new Exception('Missing valid Status on response');
845845
}
846846

847847
$codeEntry = self::query($dom, '/samlp:Response/samlp:Status/samlp:StatusCode', $statusEntry->item(0));
848-
if ($codeEntry->length == 0) {
849-
throw new Exception('Missing Status Code on response');
848+
if ($codeEntry->length != 1) {
849+
throw new Exception('Missing valid Status Code on response');
850850
}
851851
$code = $codeEntry->item(0)->getAttribute('Value');
852852
$status['code'] = $code;
853853

854+
$status['msg'] = '';
854855
$messageEntry = self::query($dom, '/samlp:Response/samlp:Status/samlp:StatusMessage', $statusEntry->item(0));
855856
if ($messageEntry->length == 0) {
856857
$subCodeEntry = self::query($dom, '/samlp:Response/samlp:Status/samlp:StatusCode/samlp:StatusCode', $statusEntry->item(0));
857-
if ($subCodeEntry->length > 0) {
858+
if ($subCodeEntry->length == 1) {
858859
$status['msg'] = $subCodeEntry->item(0)->getAttribute('Value');
859-
} else {
860-
$status['msg'] = '';
861860
}
862-
} else {
861+
} else if ($messageEntry->length == 1) {
863862
$msg = $messageEntry->item(0)->textContent;
864863
$status['msg'] = $msg;
865864
}
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)