From db3b9ed6fe795c80673fa60aef75bff8a79b5b1d Mon Sep 17 00:00:00 2001
From: Olav Morken <olav.morken@uninett.no>
Date: Mon, 13 Jul 2009 06:15:47 +0000
Subject: [PATCH] Error_Exception: Extend to support serialization.

Make the SimpleSAML_Error_Exception class suitable for serialization by
making it _not_ serialize the trace variable in the Exception class.

Serializing the trace-variable is problematic because it contains the
parameters to all function calls. This can make the exception very big.
This can also make the exception unserializable, if one of the
parameters is unserializable.

git-svn-id: https://simplesamlphp.googlecode.com/svn/trunk@1563 44740490-163a-0410-bde0-09ae8108e29a
---
 lib/SimpleSAML/Error/Exception.php | 78 +++++++++++++++++++++++-------
 1 file changed, 60 insertions(+), 18 deletions(-)

diff --git a/lib/SimpleSAML/Error/Exception.php b/lib/SimpleSAML/Error/Exception.php
index 9e1f656cd..ec631baf3 100644
--- a/lib/SimpleSAML/Error/Exception.php
+++ b/lib/SimpleSAML/Error/Exception.php
@@ -2,13 +2,26 @@
 
 /**
  * Baseclass for simpleSAML Exceptions
- * 
+ *
+ * This class tries to make sure that every exception is serializable.
+ *
  * @author Thomas Graff <thomas.graff@uninett.no>
  * @package simpleSAMLphp_base
  * @version $Id$
  */
 class SimpleSAML_Error_Exception extends Exception {
-	
+
+	/**
+	 * The backtrace for this exception.
+	 *
+	 * We need to save the backtrace, since we cannot rely on
+	 * serializing the Exception::trace-variable.
+	 *
+	 * @var string
+	 */
+	private $backtrace;
+
+
 	/**
 	 * Constructor for this error.
 	 *
@@ -17,32 +30,61 @@ class SimpleSAML_Error_Exception extends Exception {
 	 */
 	public function __construct($message, $code = 0) {
 		assert('is_string($message) || is_int($code)');
-		
+
 		parent::__construct($message, $code);
+
+		$this->backtrace = SimpleSAML_Utilities::buildBacktrace($this);
 	}
-	
-	
+
+
 	/**
-	 * Set the HTTP return code for this error.
+	 * Retrieve the backtrace.
 	 *
-	 * This should be overridden by subclasses who want a different return code than 500 Internal Server Error.
+	 * @return array  An array where each function call is a single item.
 	 */
-	protected function setHTTPCode() {
-		header('HTTP/1.0 500 Internal Server Error');
+	public function getBacktrace() {
+		return $this->backtrace;
 	}
-	
-	
+
+
 	/**
-	 * Display this error.
+	 * Replace the backtrace.
+	 *
+	 * This function is meant for subclasses which needs to replace the backtrace
+	 * of this exception, such as the SimpleSAML_Error_Unserializable class.
 	 *
-	 * This method displays a standard simpleSAMLphp error page and exits.
+	 * @param array $backtrace  The new backtrace.
 	 */
-	public function show() {
-		$this->setHTTPCode();
-		$session = SimpleSAML_Session::getInstance();
-		$e = $this;
-		SimpleSAML_Utilities::fatalError($session->getTrackID(), $this->errorCode, $e);
+	protected function setBacktrace($backtrace) {
+		assert('is_array($backtrace)');
+
+		$this->backtrace = $backtrace;
 	}
+
+
+	/**
+	 * Function for serialization.
+	 *
+	 * This function builds a list of all variables which should be serialized.
+	 * It will serialize all variables except the Exception::trace variable.
+	 *
+	 * @return array  Array with the variables which should be serialized.
+	 */
+	public function __sleep() {
+
+		$ret = array();
+
+		$ret = array_keys((array)$this);
+
+		foreach ($ret as $i => $e) {
+			if ($e === "\0Exception\0trace") {
+				unset($ret[$i]);
+			}
+		}
+
+		return $ret;
+	}
+
 }
 
 ?>
\ No newline at end of file
-- 
GitLab