From 2dd415e1f5c69ba39aba704f21f84db924b261ef Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Andreas=20=C3=85kre=20Solberg?= <andreas.solberg@uninett.no>
Date: Sun, 1 Mar 2009 12:45:07 +0000
Subject: [PATCH] add support for idp-initiated SLO with iframe.

git-svn-id: https://simplesamlphp.googlecode.com/svn/trunk@1348 44740490-163a-0410-bde0-09ae8108e29a
---
 templates/logout-iframe.php                   |  30 ++-
 www/saml2/idp/SingleLogoutService.php         |   2 +-
 www/saml2/idp/SingleLogoutServiceiFrame.php   |   6 +-
 .../idp/SingleLogoutServiceiFrameResponse.php |   1 -
 .../idp/idpInitSingleLogoutServiceiFrame.php  | 255 ++++++++++++++++++
 5 files changed, 281 insertions(+), 13 deletions(-)
 create mode 100644 www/saml2/idp/idpInitSingleLogoutServiceiFrame.php

diff --git a/templates/logout-iframe.php b/templates/logout-iframe.php
index 3c1467180..c62e75f28 100644
--- a/templates/logout-iframe.php
+++ b/templates/logout-iframe.php
@@ -57,6 +57,7 @@ function startslo() {
 function slocompletesp($entityhash) {
 	$("table#slostatustable tr#" + $entityhash).filter(".inprogress").removeClass("inprogress").addClass("completed").
 		children().fadeOut("fast").fadeIn("fast");
+	alert("complete SP " + $entityhash );
 }
 
 
@@ -97,9 +98,10 @@ function sendResponse() {
 
 	<?php
 	
-	$requestername = is_array($this->data['requesterName']) ? 
-		$this->getTranslation($this->data['requesterName']) : $this->data['requesterName'];
-
+	if (array_key_exists('requestername', $this->data)) {
+		$requestername = is_array($this->data['requesterName']) ? 
+			$this->getTranslation($this->data['requesterName']) : $this->data['requesterName'];
+	}
 	#echo('<p>' . $this->t('{logout:description}', array('%REQUESTERNAME%' => $requestername)) . '</p>');
 	
 	?>
@@ -108,11 +110,12 @@ function sendResponse() {
 
 	<?php
 	
-		echo('<div><img style="float: left; margin-right: 12px" src="/' . $this->data['baseurlpath'] . 'resources/icons/checkmark48.png" alt="Successful logout" />');
-		echo('<p style="padding-top: 16px; ">' . $this->t('{logout:loggedoutfrom}', array('%SP%' => '<strong>' .$requestername.'</strong>')) . '</p>');
-		echo('<p style="height: 0px; clear: left;"></p>');
-		echo('</div>');
-		
+		if (array_key_exists('requestername', $this->data)) {
+			echo('<div><img style="float: left; margin-right: 12px" src="/' . $this->data['baseurlpath'] . 'resources/icons/checkmark48.png" alt="Successful logout" />');
+			echo('<p style="padding-top: 16px; ">' . $this->t('{logout:loggedoutfrom}', array('%SP%' => '<strong>' .$requestername.'</strong>')) . '</p>');
+			echo('<p style="height: 0px; clear: left;"></p>');
+			echo('</div>');
+		}
 
 		echo('<div style="margin-top: 3em; clear: both">');
 		echo('<p style="margin-bottom: .5em">' . $this->t('{logout:also_from}') . '</p>');
@@ -187,7 +190,16 @@ function sendResponse() {
 			<?php echo $this->t('{logout:logout_all_question}'); ?> <br />
 		</p>
 		<input type="button" id="ok" name="ok" value="<?php echo $this->t('{logout:logout_all}'); ?>" />
-		<input type="button" id="cancel" name="cancel" value="<?php echo $this->t('{logout:logout_only}', array('%SP%' => $requestername)); ?>" />
+		<?php
+			
+			if (array_key_exists('requestername', $this->data)) {
+				echo '<input type="button" id="cancel" name="cancel" value="' . $this->t('{logout:logout_only}', array('%SP%' => $requestername)) . '" />';
+			} else {
+				echo '<input type="button" id="cancel" name="cancel" value="' . $this->t('{logout:no}') . '" />';
+			}
+			
+		?>
+		
 		
 
 		<p id="incapablesps" >
diff --git a/www/saml2/idp/SingleLogoutService.php b/www/saml2/idp/SingleLogoutService.php
index 22e2caa9f..04809ac7b 100644
--- a/www/saml2/idp/SingleLogoutService.php
+++ b/www/saml2/idp/SingleLogoutService.php
@@ -115,7 +115,7 @@ if (isset($_GET['SAMLRequest'])) {
 	$responder = $metadata->getMetaDataCurrentEntityID('saml20-idp-hosted');
 	
 	
-	SimpleSAML_Logger::info('SAML2.0 - IdP.SingleLogoutService: got Logoutrequest from ' . $logoutrequest->getIssuer());
+	SimpleSAML_Logger::info('SAML2.0 - IdP.SingleLogoutService: got Logoutrequest from ' . $requester);
 	SimpleSAML_Logger::stats('saml20-idp-SLO spinit ' . $requester . ' ' . $responder);
 	
 	/* Check if we have a valid session. */
diff --git a/www/saml2/idp/SingleLogoutServiceiFrame.php b/www/saml2/idp/SingleLogoutServiceiFrame.php
index 2985f746f..625a45ae1 100644
--- a/www/saml2/idp/SingleLogoutServiceiFrame.php
+++ b/www/saml2/idp/SingleLogoutServiceiFrame.php
@@ -17,6 +17,8 @@ $session = SimpleSAML_Session::getInstance();
 
 SimpleSAML_Logger::info('SAML2.0 - IdP.SingleLogoutServiceiFrame: Accessing SAML 2.0 IdP endpoint SingleLogoutService (iFrame version)');
 
+SimpleSAML_Logger::debug('Initially; ' . join(',', $session->get_sp_list(SimpleSAML_Session::STATE_ONLINE)));
+
 if (!$config->getValue('enable.saml20-idp', false))
 	SimpleSAML_Utilities::fatalError(isset($session) ? $session->getTrackID() : null, 'NOACCESS');
 
@@ -103,8 +105,8 @@ function updateslostatus() {
 	foreach ($templistofsps AS $spentityid) {
 		if (!empty($_COOKIE['spstate-' . sha1($spentityid)])) $listofsps[] = $spentityid;
 	}
-	SimpleSAML_Logger::debug('SAML2.0 - IdP.SingleLogoutServiceiFrame: templistofsps ' . var_export($templistofsps, TRUE));
-	SimpleSAML_Logger::debug('SAML2.0 - IdP.SingleLogoutServiceiFrame:     listofsps ' . var_export($listofsps, TRUE));
+	SimpleSAML_Logger::debug('SAML2.0 - IdP.SingleLogoutServiceiFrame: templistofsps ' . join(',', $templistofsps));
+	SimpleSAML_Logger::debug('SAML2.0 - IdP.SingleLogoutServiceiFrame:     listofsps ' . join(',', $listofsps));
 
 
 	// Using template object to be able to translate name of service provider.
diff --git a/www/saml2/idp/SingleLogoutServiceiFrameResponse.php b/www/saml2/idp/SingleLogoutServiceiFrameResponse.php
index 545039d93..a89a450fd 100644
--- a/www/saml2/idp/SingleLogoutServiceiFrameResponse.php
+++ b/www/saml2/idp/SingleLogoutServiceiFrameResponse.php
@@ -70,7 +70,6 @@ if (isset($_GET['SAMLResponse'])) {
 
 	SimpleSAML_Utilities::fatalError($session->getTrackID(), 'SLOSERVICEPARAMS', 
 		new Exception('No valid SAMLResponse found? Probably some error in remote partys metadata that sends something to this endpoint that is not SAML LogoutResponses') );
-#	echo 'Not set: SAMLResponse';
 }
 
 ?>
\ No newline at end of file
diff --git a/www/saml2/idp/idpInitSingleLogoutServiceiFrame.php b/www/saml2/idp/idpInitSingleLogoutServiceiFrame.php
new file mode 100644
index 000000000..682b4eebf
--- /dev/null
+++ b/www/saml2/idp/idpInitSingleLogoutServiceiFrame.php
@@ -0,0 +1,255 @@
+<?php
+
+/**
+ * IdP Initiated Single Log-Out. Requires one parameter: RelayState.
+ *
+ * @author Andreas Ă…kre Solberg, UNINETT AS. <andreas.solberg@uninett.no>
+ * @package simpleSAMLphp
+ * @version $Id$
+ */
+
+require_once('../../_include.php');
+
+$config = SimpleSAML_Configuration::getInstance();
+$metadata = SimpleSAML_Metadata_MetaDataStorageHandler::getMetadataHandler();
+$session = SimpleSAML_Session::getInstance();
+
+SimpleSAML_Logger::info('SAML2.0 - IdP.idpInitSingleLogoutServiceiFrame: Accessing SAML 2.0 IdP endpoint SingleLogoutService (iFrame version)');
+
+SimpleSAML_Logger::debug('Initially; ' . join(',', $session->get_sp_list(SimpleSAML_Session::STATE_ONLINE)));
+
+if (!$config->getValue('enable.saml20-idp', false))
+	SimpleSAML_Utilities::fatalError(isset($session) ? $session->getTrackID() : null, 'NOACCESS');
+
+try {
+	$idpentityid = $metadata->getMetaDataCurrentEntityID('saml20-idp-hosted');
+} catch (Exception $exception) {
+	SimpleSAML_Utilities::fatalError($session->getTrackID(), 'METADATA', $exception);
+}
+
+SimpleSAML_Logger::debug('SAML2.0 - IdP.SingleLogoutServiceiFrame: Got IdP entity id: ' . $idpentityid);
+
+
+
+$logouttype = 'traditional';
+$idpmeta = $metadata->getMetaDataCurrent('saml20-idp-hosted');
+if (array_key_exists('logouttype', $idpmeta)) $logouttype = $idpmeta['logouttype'];
+
+if ($logouttype !== 'iframe') 
+	SimpleSAML_Utilities::fatalError($session->getTrackID(), 'NOACCESS', new Exception('This IdP is configured to use logout type [' . $logouttype . '], but this endpoint is only available for IdP using logout type [iframe]'));
+
+
+
+
+
+
+
+/**
+ * This function retrieves the logout info with the given ID.
+ *
+ * @param $id  The identifier of the logout info.
+ */
+function fetchLogoutInfo($id) {
+	global $session;
+	global $logoutInfo;
+
+	$logoutInfo = $session->getData('idplogoutresponsedata', $id);
+
+	if($logoutInfo === NULL) {
+		SimpleSAML_Logger::warning('SAML2.0 - IdP.SingleLogoutService: Lost logout information.');
+	}
+}
+
+
+/**
+ * This function saves the logout info with the given ID.
+ *
+ * @param $id  The identifier the logout info should be saved with.
+ */
+function saveLogoutInfo($id) {
+	global $session;
+	global $logoutInfo;
+
+	$session->setData('idplogoutresponsedata', $id, $logoutInfo);
+}
+
+
+// Include XAJAX definition.
+require_once(SimpleSAML_Utilities::resolvePath('libextinc') . '/xajax/xajax.inc.php');
+
+
+
+/*
+ * This function is called via AJAX and will send LogoutRequest to one single SP by
+ * sending a LogoutRequest using HTTP-REDIRECT
+ */
+function updateslostatus() {
+
+	SimpleSAML_Logger::info('SAML2.0 - IdP.SingleLogoutServiceiFrame: Accessing SAML 2.0 IdP endpoint SingleLogoutService (iFrame version) within updateslostatus() ');
+
+	$config = SimpleSAML_Configuration::getInstance();
+	$metadata = SimpleSAML_Metadata_MetaDataStorageHandler::getMetadataHandler();
+	$session = SimpleSAML_Session::getInstance();
+
+	$idpentityid = $metadata->getMetaDataCurrentEntityID('saml20-idp-hosted');
+	
+	$templistofsps = $session->get_sp_list(SimpleSAML_Session::STATE_ONLINE);
+	$listofsps = array();
+	foreach ($templistofsps AS $spentityid) {
+		if (!empty($_COOKIE['spstate-' . sha1($spentityid)])) $listofsps[] = $spentityid;
+	}
+	SimpleSAML_Logger::debug('SAML2.0 - IdP.SingleLogoutServiceiFrame: templistofsps ' . join(',', $templistofsps));
+	SimpleSAML_Logger::debug('SAML2.0 - IdP.SingleLogoutServiceiFrame:     listofsps ' . join(',', $listofsps));
+
+
+	// Using template object to be able to translate name of service provider.
+	$t = new SimpleSAML_XHTML_Template($config, 'logout-iframe.php');
+
+    // Instantiate the xajaxResponse object
+    $objResponse = new xajaxResponse();
+
+	foreach ($listofsps AS $spentityid) {
+
+		SimpleSAML_Logger::debug('SAML2.0 - IdP.SingleLogoutServiceiFrame: Completed ' . $spentityid);
+		
+		// add a command to the response to assign the innerHTML attribute of
+		// the element with id="SomeElementId" to whatever the new content is
+		
+		$spmetadata = $metadata->getMetaData($spentityid, 'saml20-sp-remote');
+		$name = array_key_exists('name', $spmetadata) ? $spmetadata['name'] : $spentityid;
+		
+		$spname = is_array($name) ? $t->getTranslation($name) : $name;
+		
+		$objResponse->addScriptCall('slocompletesp', 'e' . sha1($spentityid));
+
+	}
+	
+	if (count($templistofsps) === count($listofsps)) {
+
+		$templistofsps = $session->get_sp_list(SimpleSAML_Session::STATE_ONLINE);
+		foreach ($templistofsps AS $spentityid) {
+			$session->set_sp_logout_completed($spentityid);
+			setcookie('spstate-' . sha1($spentityid) , '', time() - 3600); // Delete cookie
+		}
+
+		$objResponse->addScriptCall('slocompleted');
+
+		/**
+		 * Clean up session object to save storage.
+		 */
+		if ($config->getValue('debug', false)) 
+			SimpleSAML_Logger::info('SAML2.0 - IdP.SingleLogoutService: Session Size before cleaning: ' . $session->getSize());
+			
+		$session->clean();
+		
+		if ($config->getValue('debug', false)) 
+			SimpleSAML_Logger::info('SAML2.0 - IdP.SingleLogoutService: Session Size after cleaning: ' . $session->getSize());
+
+	} else {
+		SimpleSAML_Logger::debug('SAML2.0 - sp_logout_completed FALSE');
+	}
+    
+    //return the  xajaxResponse object
+    return $objResponse;
+}
+
+
+
+$xajax = new xajax();
+$xajax->registerFunction("updateslostatus");
+$xajax->processRequests();
+
+
+
+
+/**
+ * Which URL to send the user to after logout?
+ */
+$relayState = NULL;
+if (array_key_exists('RelayState', $_REQUEST)) $relayState = $_REQUEST['RelayState'];
+
+// Do logout from the IdP
+$session->doLogout();
+
+// Debug entries in the log about what services the user is logged into.
+$session->dump_sp_sessions();
+
+
+
+/*
+ * Generate a list of all service providers, and create a LogoutRequest message for all these SPs.
+ */
+$listofsps = $session->get_sp_list();
+$sparray = array();
+$sparrayNoLogout = array();
+foreach ($listofsps AS $spentityid) {
+
+	// ($issuer, $receiver, $nameid, $nameidformat, $sessionindex, $mode) {
+	$nameId = $session->getSessionNameId('saml20-sp-remote', $spentityid);
+	if($nameId === NULL) {
+		$nameId = $session->getNameID();
+	}
+
+
+	$spmetadata = $metadata->getMetaData($spentityid, 'saml20-sp-remote');
+	$name = array_key_exists('name', $spmetadata) ? $spmetadata['name'] : $spentityid;
+
+	try {	
+		$lr = new SimpleSAML_XML_SAML20_LogoutRequest($config, $metadata);
+		$req = $lr->generate($idpentityid, $spentityid, $nameId, $session->getSessionIndex(), 'IdP');
+		$httpredirect = new SimpleSAML_Bindings_SAML20_HTTPRedirect($config, $metadata);
+		// $request, $localentityid, $remoteentityid, $relayState = null, $endpoint = 'SingleSignOnService', $direction = 'SAMLRequest', $mode = 'SP'
+		$url = $httpredirect->getRedirectURL($req, $idpentityid, $spentityid, NULL, 'SingleLogoutService', 'SAMLRequest', 'IdP');
+
+	
+		$sparray[$spentityid] = array('url' => $url, 'name' => $name);
+		
+	} catch (Exception $e) {
+		
+		$sparrayNoLogout[$spentityid] = array('name' => $name);
+		
+	}
+
+}
+
+
+SimpleSAML_Logger::debug('SAML2.0 - SP Counter. other SPs with SLO support (' . count($sparray) . ')  without SLO support (' . count($sparrayNoLogout) . ')');
+
+
+#print_r($sparray);
+
+
+
+
+/*
+ * If the user is not logged into any other SPs.
+ */
+if (count($sparray) === 0) {
+	SimpleSAML_Utilities::redirect($relayState);
+	exit;
+} 
+
+
+$et = new SimpleSAML_XHTML_Template($config, 'logout-iframe.php');
+
+$et->data['header'] = 'Logout';
+$et->data['sparray'] = $sparray;
+$et->data['sparrayNoLogout'] = $sparrayNoLogout;
+
+$et->data['logoutresponse'] = $relayState;
+$et->data['xajax'] = $xajax;
+
+#$et->data['idpInitRelayState'] = $relayState;
+
+# $et->data['requesterName'] = $spname;
+
+$et->data['head'] = $xajax->getJavascript();
+
+$et->show();
+
+exit(0);
+
+
+
+
+?>
\ No newline at end of file
-- 
GitLab