2008-05-30 11:40:08 +00:00
< ? php
/**
* Framework debugging and PHP error - handling class
*
* Provides enhanced logging , stack traces , and rendering debug views
*
* PHP versions 4 and 5
*
2009-05-01 14:05:46 -07:00
* CakePHP ( tm ) : Rapid Development Framework ( http :// cakephp . org )
* Copyright 2005 - 2009 , Cake Software Foundation , Inc . ( http :// cakefoundation . org )
2008-05-30 11:40:08 +00:00
*
* Licensed under The MIT License
* Redistributions of files must retain the above copyright notice .
*
2009-05-01 14:05:46 -07:00
* @ copyright Copyright 2005 - 2009 , Cake Software Foundation , Inc . ( http :// cakefoundation . org )
* @ link http :// cakephp . org
2008-10-30 17:30:26 +00:00
* @ package cake
* @ subpackage cake . cake . libs
* @ since CakePHP ( tm ) v 1.2 . 4560
2009-05-01 14:05:46 -07:00
* @ license MIT License ( http :// www . opensource . org / licenses / mit - license . php )
2008-05-30 11:40:08 +00:00
*/
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
* Included libraries .
*
*/
2009-05-01 14:05:46 -07:00
if ( ! class_exists ( 'Object' )) {
require_once LIBS . 'object.php' ;
}
if ( ! class_exists ( 'CakeLog' )) {
require_once LIBS . 'cake_log.php' ;
}
if ( ! class_exists ( 'String' )) {
require_once LIBS . 'string.php' ;
}
2008-05-30 11:40:08 +00:00
/**
* Provide custom logging and error handling .
*
* Debugger overrides PHP ' s default error handling to provide stack traces and enhanced logging
*
2008-10-30 17:30:26 +00:00
* @ package cake
* @ subpackage cake . cake . libs
2008-10-31 20:17:26 +00:00
* @ link http :// book . cakephp . org / view / 460 / Using - the - Debugger - Class
2008-05-30 11:40:08 +00:00
*/
class Debugger extends Object {
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 20:17:26 +00:00
* A list of errors generated by the application .
2008-05-30 11:40:08 +00:00
*
* @ var array
* @ access public
*/
var $errors = array ();
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 20:17:26 +00:00
* Contains the base URL for error code documentation .
2008-05-30 11:40:08 +00:00
*
* @ var string
* @ access public
*/
var $helpPath = null ;
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 20:17:26 +00:00
* The current output format .
2008-05-30 11:40:08 +00:00
*
* @ var string
2008-12-07 20:44:00 +00:00
* @ access protected
2008-05-30 11:40:08 +00:00
*/
2008-12-07 20:44:00 +00:00
var $_outputFormat = 'js' ;
2009-05-01 14:05:46 -07:00
/**
* Templates used when generating trace or error strings . Can be global or indexed by the format
* value used in $_outputFormat .
*
* @ var string
* @ access protected
*/
var $_templates = array (
'log' => array (
'trace' => '{:reference} - {:path}, line {:line}' ,
'error' => " { :error} ( { :code}): { :description} in [ { :file}, line { :line}] "
),
'js' => array (
'error' => '' ,
'info' => '' ,
'trace' => '<pre class="stack-trace">{:trace}</pre>' ,
'code' => '' ,
'context' => '' ,
'links' => array ()
),
'html' => array (
'trace' => '<pre class="cake-debug trace"><b>Trace</b> <p>{:trace}</p></pre>' ,
'context' => '<pre class="cake-debug context"><b>Context</b> <p>{:context}</p></pre>'
),
'txt' => array (
'error' => " { :error}: { :code} :: { :description} on line { :line} of { :path} \n { :info} " ,
'context' => " Context: \n { :context} \n " ,
'trace' => " Trace: \n { :trace} \n " ,
'code' => '' ,
'info' => ''
),
'base' => array (
'traceLine' => '{:reference} - {:path}, line {:line}'
)
);
2008-05-30 11:40:08 +00:00
/**
2008-10-31 20:17:26 +00:00
* Holds current output data when outputFormat is false .
2008-05-30 11:40:08 +00:00
*
* @ var string
* @ access private
*/
2009-05-01 14:05:46 -07:00
var $_data = array ();
2008-05-30 11:40:08 +00:00
/**
2008-10-31 20:17:26 +00:00
* Constructor .
2008-05-30 11:40:08 +00:00
*
*/
function __construct () {
$docRef = ini_get ( 'docref_root' );
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
if ( empty ( $docRef )) {
ini_set ( 'docref_root' , 'http://php.net/' );
}
if ( ! defined ( 'E_RECOVERABLE_ERROR' )) {
define ( 'E_RECOVERABLE_ERROR' , 4096 );
}
2009-05-01 14:05:46 -07:00
if ( ! defined ( 'E_DEPRECATED' )) {
define ( 'E_DEPRECATED' , 8192 );
}
2009-07-25 10:27:17 -07:00
$e = '<pre class="cake-debug">' ;
$e .= '<a href="javascript:void(0);" onclick="document.getElementById(\'{:id}-trace\')' ;
2009-05-01 14:05:46 -07:00
$e .= '.style.display = (document.getElementById(\'{:id}-trace\').style.display == ' ;
$e .= '\'none\' ? \'\' : \'none\');"><b>{:error}</b> ({:code})</a>: {:description} ' ;
$e .= '[<b>{:path}</b>, line <b>{:line}</b>]' ;
$e .= '<div id="{:id}-trace" class="cake-stack-trace" style="display: none;">' ;
$e .= '{:links}{:info}</div>' ;
2009-07-25 10:27:17 -07:00
$e .= '</pre>' ;
2009-05-01 14:05:46 -07:00
$this -> _templates [ 'js' ][ 'error' ] = $e ;
$t = '<div id="{:id}-trace" class="cake-stack-trace" style="display: none;">' ;
$t .= '{:context}{:code}{:trace}</div>' ;
$this -> _templates [ 'js' ][ 'info' ] = $t ;
$links = array ();
$link = '<a href="javascript:void(0);" onclick="document.getElementById(\'{:id}-code\')' ;
$link .= '.style.display = (document.getElementById(\'{:id}-code\').style.display == ' ;
$link .= '\'none\' ? \'\' : \'none\')">Code</a>' ;
$links [ 'code' ] = $link ;
$link = '<a href="javascript:void(0);" onclick="document.getElementById(\'{:id}-context\')' ;
$link .= '.style.display = (document.getElementById(\'{:id}-context\').style.display == ' ;
$link .= '\'none\' ? \'\' : \'none\')">Context</a>' ;
$links [ 'context' ] = $link ;
$links [ 'help' ] = '<a href="{:helpPath}{:code}" target="_blank">Help</a>' ;
$this -> _templates [ 'js' ][ 'links' ] = $links ;
$this -> _templates [ 'js' ][ 'context' ] = '<pre id="{:id}-context" class="cake-context" ' ;
$this -> _templates [ 'js' ][ 'context' ] .= 'style="display: none;">{:context}</pre>' ;
$this -> _templates [ 'js' ][ 'code' ] = '<div id="{:id}-code" class="cake-code-dump" ' ;
$this -> _templates [ 'js' ][ 'code' ] .= 'style="display: none;"><pre>{:code}</pre></div>' ;
2009-07-25 10:27:17 -07:00
$e = '<pre class="cake-debug"><b>{:error}</b> ({:code}) : {:description} ' ;
2009-05-01 14:05:46 -07:00
$e .= '[<b>{:path}</b>, line <b>{:line}]</b></pre>' ;
$this -> _templates [ 'html' ][ 'error' ] = $e ;
$this -> _templates [ 'html' ][ 'context' ] = '<pre class="cake-debug context"><b>Context</b> ' ;
$this -> _templates [ 'html' ][ 'context' ] .= '<p>{:context}</p></pre>' ;
2008-05-30 11:40:08 +00:00
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 20:17:26 +00:00
* Returns a reference to the Debugger singleton object instance .
2008-05-30 11:40:08 +00:00
*
* @ return object
* @ access public
2008-09-25 16:49:56 +00:00
* @ static
2008-05-30 11:40:08 +00:00
*/
2008-12-07 20:44:00 +00:00
function & getInstance ( $class = null ) {
2008-05-30 11:40:08 +00:00
static $instance = array ();
2008-12-07 20:44:00 +00:00
if ( ! empty ( $class )) {
if ( ! $instance || strtolower ( $class ) != strtolower ( get_class ( $instance [ 0 ]))) {
$instance [ 0 ] = & new $class ();
if ( Configure :: read () > 0 ) {
Configure :: version (); // Make sure the core config is loaded
$instance [ 0 ] -> helpPath = Configure :: read ( 'Cake.Debugger.HelpPath' );
}
}
}
2008-05-30 11:40:08 +00:00
2008-09-12 05:11:34 +00:00
if ( ! $instance ) {
2008-05-30 11:40:08 +00:00
$instance [ 0 ] =& new Debugger ();
if ( Configure :: read () > 0 ) {
Configure :: version (); // Make sure the core config is loaded
$instance [ 0 ] -> helpPath = Configure :: read ( 'Cake.Debugger.HelpPath' );
}
}
return $instance [ 0 ];
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 20:17:26 +00:00
* Formats and outputs the contents of the supplied variable .
2008-10-12 03:52:24 +00:00
*
2008-09-25 16:49:56 +00:00
* @ param $var mixed the variable to dump
* @ return void
* @ see exportVar
* @ access public
* @ static
2008-10-31 20:17:26 +00:00
* @ link http :// book . cakephp . org / view / 460 / Using - the - Debugger - Class
2008-05-30 11:40:08 +00:00
*/
function dump ( $var ) {
$_this = Debugger :: getInstance ();
pr ( $_this -> exportVar ( $var ));
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 20:17:26 +00:00
* Creates a detailed stack trace log at the time of invocation , much like dump ()
* but to debug . log .
2008-10-12 03:52:24 +00:00
*
2008-09-25 16:49:56 +00:00
* @ param $var mixed Variable or content to log
* @ param $level int type of log to use . Defaults to LOG_DEBUG
* @ return void
* @ static
2008-10-31 20:17:26 +00:00
* @ link http :// book . cakephp . org / view / 460 / Using - the - Debugger - Class
2008-09-25 16:49:56 +00:00
*/
2008-06-19 14:23:29 +00:00
function log ( $var , $level = LOG_DEBUG ) {
2008-05-30 11:40:08 +00:00
$_this = Debugger :: getInstance ();
$trace = $_this -> trace ( array ( 'start' => 1 , 'depth' => 2 , 'format' => 'array' ));
$source = null ;
if ( is_object ( $trace [ 0 ][ 'object' ]) && isset ( $trace [ 0 ][ 'object' ] -> _reporter -> _test_stack )) {
$stack = $trace [ 0 ][ 'object' ] -> _reporter -> _test_stack ;
2009-03-23 18:59:37 +00:00
$source = sprintf ( '[%1$s, %3$s::%2$s()]' . " \n " ,
array_shift ( $stack ), array_pop ( $stack ), array_pop ( $stack ));
2008-05-30 11:40:08 +00:00
}
CakeLog :: write ( $level , $source . $_this -> exportVar ( $var ));
}
/**
2008-10-31 20:17:26 +00:00
* Overrides PHP ' s default error handling .
2008-05-30 11:40:08 +00:00
*
* @ param integer $code Code of error
* @ param string $description Error description
* @ param string $file File on which error occurred
* @ param integer $line Line that triggered the error
* @ param array $context Context
* @ return boolean true if error was handled
* @ access public
*/
function handleError ( $code , $description , $file = null , $line = null , $context = null ) {
2009-07-31 11:48:30 -04:00
if ( error_reporting () == 0 || $code === 2048 || $code === 8192 ) {
2008-05-30 11:40:08 +00:00
return ;
}
$_this = Debugger :: getInstance ();
if ( empty ( $file )) {
$file = '[internal]' ;
}
if ( empty ( $line )) {
$line = '??' ;
}
2009-05-01 14:05:46 -07:00
$path = $_this -> trimPath ( $file );
2008-05-30 11:40:08 +00:00
$info = compact ( 'code' , 'description' , 'file' , 'line' );
if ( ! in_array ( $info , $_this -> errors )) {
$_this -> errors [] = $info ;
} else {
return ;
}
switch ( $code ) {
case E_PARSE :
case E_ERROR :
case E_CORE_ERROR :
case E_COMPILE_ERROR :
case E_USER_ERROR :
$error = 'Fatal Error' ;
$level = LOG_ERROR ;
break ;
case E_WARNING :
case E_USER_WARNING :
case E_COMPILE_WARNING :
case E_RECOVERABLE_ERROR :
$error = 'Warning' ;
$level = LOG_WARNING ;
break ;
case E_NOTICE :
case E_USER_NOTICE :
$error = 'Notice' ;
$level = LOG_NOTICE ;
break ;
default :
2009-09-10 12:32:21 -03:00
return ;
2008-05-30 11:40:08 +00:00
break ;
}
$helpCode = null ;
if ( ! empty ( $_this -> helpPath ) && preg_match ( '/.*\[([0-9]+)\]$/' , $description , $codes )) {
if ( isset ( $codes [ 1 ])) {
2009-05-01 14:05:46 -07:00
$helpID = $codes [ 1 ];
2008-05-30 11:40:08 +00:00
$description = trim ( preg_replace ( '/\[[0-9]+\]$/' , '' , $description ));
}
}
2009-05-01 14:05:46 -07:00
$data = compact (
'level' , 'error' , 'code' , 'helpID' , 'description' , 'file' , 'path' , 'line' , 'context'
);
echo $_this -> _output ( $data );
2008-05-30 11:40:08 +00:00
if ( Configure :: read ( 'log' )) {
2009-05-01 14:05:46 -07:00
$tpl = $_this -> _templates [ 'log' ][ 'error' ];
$options = array ( 'before' => '{:' , 'after' => '}' );
CakeLog :: write ( $level , String :: insert ( $tpl , $data , $options ));
2008-05-30 11:40:08 +00:00
}
if ( $error == 'Fatal Error' ) {
die ();
}
return true ;
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 20:17:26 +00:00
* Outputs a stack trace based on the supplied options .
2008-05-30 11:40:08 +00:00
*
* @ param array $options Format for outputting stack trace
* @ return string Formatted stack trace
* @ access public
2008-09-25 16:49:56 +00:00
* @ static
2008-10-31 20:17:26 +00:00
* @ link http :// book . cakephp . org / view / 460 / Using - the - Debugger - Class
2008-05-30 11:40:08 +00:00
*/
function trace ( $options = array ()) {
2009-05-01 14:05:46 -07:00
$_this = Debugger :: getInstance ();
$defaults = array (
'depth' => 999 ,
'format' => $_this -> _outputFormat ,
'args' => false ,
'start' => 0 ,
'scope' => null ,
'exclude' => null
2008-05-30 11:40:08 +00:00
);
2009-05-01 14:05:46 -07:00
$options += $defaults ;
2008-05-30 11:40:08 +00:00
$backtrace = debug_backtrace ();
2008-09-18 03:09:19 +00:00
$count = count ( $backtrace );
2009-05-01 14:05:46 -07:00
$back = array ();
$_trace = array (
'line' => '??' ,
'file' => '[internal]' ,
'class' => null ,
'function' => '[main]'
);
2008-05-30 11:40:08 +00:00
2008-09-18 03:09:19 +00:00
for ( $i = $options [ 'start' ]; $i < $count && $i < $options [ 'depth' ]; $i ++ ) {
2009-05-01 14:05:46 -07:00
$trace = array_merge ( array ( 'file' => '[internal]' , 'line' => '??' ), $backtrace [ $i ]);
2008-05-30 11:40:08 +00:00
if ( isset ( $backtrace [ $i + 1 ])) {
2009-05-01 14:05:46 -07:00
$next = array_merge ( $_trace , $backtrace [ $i + 1 ]);
$reference = $next [ 'function' ];
2008-05-30 11:40:08 +00:00
if ( ! empty ( $next [ 'class' ])) {
2009-05-01 14:05:46 -07:00
$reference = $next [ 'class' ] . '::' . $reference . '(' ;
2008-05-30 11:40:08 +00:00
if ( $options [ 'args' ] && isset ( $next [ 'args' ])) {
$args = array ();
foreach ( $next [ 'args' ] as $arg ) {
$args [] = Debugger :: exportVar ( $arg );
}
2009-05-01 14:05:46 -07:00
$reference .= join ( ', ' , $args );
2008-05-30 11:40:08 +00:00
}
2009-05-01 14:05:46 -07:00
$reference .= ')' ;
2008-05-30 11:40:08 +00:00
}
} else {
2009-05-01 14:05:46 -07:00
$reference = '[main]' ;
2008-05-30 11:40:08 +00:00
}
2009-05-01 14:05:46 -07:00
if ( in_array ( $reference , array ( 'call_user_func_array' , 'trigger_error' ))) {
2008-05-30 11:40:08 +00:00
continue ;
}
if ( $options [ 'format' ] == 'points' && $trace [ 'file' ] != '[internal]' ) {
$back [] = array ( 'file' => $trace [ 'file' ], 'line' => $trace [ 'line' ]);
2009-05-01 14:05:46 -07:00
} elseif ( $options [ 'format' ] == 'array' ) {
2008-05-30 11:40:08 +00:00
$back [] = $trace ;
2009-05-01 14:05:46 -07:00
} else {
if ( isset ( $_this -> _templates [ $options [ 'format' ]][ 'traceLine' ])) {
$tpl = $_this -> _templates [ $options [ 'format' ]][ 'traceLine' ];
} else {
$tpl = $_this -> _templates [ 'base' ][ 'traceLine' ];
}
$trace [ 'path' ] = Debugger :: trimPath ( $trace [ 'file' ]);
$trace [ 'reference' ] = $reference ;
unset ( $trace [ 'object' ], $trace [ 'args' ]);
$back [] = String :: insert ( $tpl , $trace , array ( 'before' => '{:' , 'after' => '}' ));
2008-05-30 11:40:08 +00:00
}
}
if ( $options [ 'format' ] == 'array' || $options [ 'format' ] == 'points' ) {
return $back ;
}
return join ( " \n " , $back );
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
* Shortens file paths by replacing the application base path with 'APP' , and the CakePHP core
2008-10-31 20:17:26 +00:00
* path with 'CORE' .
2008-05-30 11:40:08 +00:00
*
* @ param string $path Path to shorten
* @ return string Normalized path
* @ access public
2008-09-25 16:49:56 +00:00
* @ static
2008-05-30 11:40:08 +00:00
*/
function trimPath ( $path ) {
if ( ! defined ( 'CAKE_CORE_INCLUDE_PATH' ) || ! defined ( 'APP' )) {
return $path ;
}
if ( strpos ( $path , APP ) === 0 ) {
return str_replace ( APP , 'APP' . DS , $path );
} elseif ( strpos ( $path , CAKE_CORE_INCLUDE_PATH ) === 0 ) {
return str_replace ( CAKE_CORE_INCLUDE_PATH , 'CORE' , $path );
} elseif ( strpos ( $path , ROOT ) === 0 ) {
return str_replace ( ROOT , 'ROOT' , $path );
}
$corePaths = Configure :: corePaths ( 'cake' );
foreach ( $corePaths as $corePath ) {
if ( strpos ( $path , $corePath ) === 0 ) {
return str_replace ( $corePath , 'CORE' . DS . 'cake' . DS , $path );
}
}
return $path ;
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
* Grabs an excerpt from a file and highlights a given line of code
*
* @ param string $file Absolute path to a PHP file
* @ param integer $line Line number to highlight
* @ param integer $context Number of lines of context to extract above and below $line
* @ return array Set of lines highlighted
* @ access public
2008-09-25 16:49:56 +00:00
* @ static
2008-10-31 20:17:26 +00:00
* @ link http :// book . cakephp . org / view / 460 / Using - the - Debugger - Class
2008-05-30 11:40:08 +00:00
*/
function excerpt ( $file , $line , $context = 2 ) {
$data = $lines = array ();
2008-10-18 01:23:33 +00:00
if ( ! file_exists ( $file )) {
return array ();
}
2008-05-30 11:40:08 +00:00
$data = @ explode ( " \n " , file_get_contents ( $file ));
if ( empty ( $data ) || ! isset ( $data [ $line ])) {
return ;
}
for ( $i = $line - ( $context + 1 ); $i < $line + $context ; $i ++ ) {
if ( ! isset ( $data [ $i ])) {
continue ;
}
$string = str_replace ( array ( " \r \n " , " \n " ), " " , highlight_string ( $data [ $i ], true ));
if ( $i == $line ) {
$lines [] = '<span class="code-highlight">' . $string . '</span>' ;
} else {
$lines [] = $string ;
}
}
return $lines ;
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 20:17:26 +00:00
* Converts a variable to a string for debug output .
2008-05-30 11:40:08 +00:00
*
* @ param string $var Variable to convert
* @ return string Variable as a formatted string
* @ access public
2008-09-25 16:49:56 +00:00
* @ static
2008-10-31 20:17:26 +00:00
* @ link http :// book . cakephp . org / view / 460 / Using - the - Debugger - Class
2008-05-30 11:40:08 +00:00
*/
function exportVar ( $var , $recursion = 0 ) {
$_this = Debugger :: getInstance ();
2008-10-23 00:10:44 +00:00
switch ( strtolower ( gettype ( $var ))) {
2008-05-30 11:40:08 +00:00
case 'boolean' :
2008-07-30 20:34:01 +00:00
return ( $var ) ? 'true' : 'false' ;
2008-05-30 11:40:08 +00:00
break ;
case 'integer' :
case 'double' :
return $var ;
break ;
case 'string' :
if ( trim ( $var ) == " " ) {
return '""' ;
}
return '"' . h ( $var ) . '"' ;
break ;
case 'object' :
return get_class ( $var ) . " \n " . $_this -> __object ( $var );
case 'array' :
$out = " array( " ;
$vars = array ();
foreach ( $var as $key => $val ) {
if ( $recursion >= 0 ) {
if ( is_numeric ( $key )) {
$vars [] = " \n \t " . $_this -> exportVar ( $val , $recursion - 1 );
} else {
$vars [] = " \n \t " . $_this -> exportVar ( $key , $recursion - 1 )
. ' => ' . $_this -> exportVar ( $val , $recursion - 1 );
}
}
}
$n = null ;
if ( count ( $vars ) > 0 ) {
$n = " \n " ;
}
return $out . join ( " , " , $vars ) . " { $n } ) " ;
break ;
case 'resource' :
return strtolower ( gettype ( $var ));
break ;
case 'null' :
return 'null' ;
break ;
}
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 20:17:26 +00:00
* Handles object to string conversion .
2008-05-30 11:40:08 +00:00
*
* @ param string $var Object to convert
2008-09-25 16:49:56 +00:00
* @ return string
2008-05-30 11:40:08 +00:00
* @ access private
2008-10-31 20:17:26 +00:00
* @ see Debugger : exportVar ()
2008-05-30 11:40:08 +00:00
*/
function __object ( $var ) {
$out = array ();
2008-10-23 00:10:44 +00:00
if ( is_object ( $var )) {
2008-05-30 11:40:08 +00:00
$className = get_class ( $var );
$objectVars = get_object_vars ( $var );
2008-10-23 00:10:44 +00:00
foreach ( $objectVars as $key => $value ) {
if ( is_object ( $value )) {
2008-05-30 11:40:08 +00:00
$value = get_class ( $value ) . ' object' ;
} elseif ( is_array ( $value )) {
$value = 'array' ;
} elseif ( $value === null ) {
$value = 'NULL' ;
} elseif ( in_array ( gettype ( $value ), array ( 'boolean' , 'integer' , 'double' , 'string' , 'array' , 'resource' ))) {
$value = Debugger :: exportVar ( $value );
}
$out [] = " $className :: $ $key = " . $value ;
}
}
return join ( " \n " , $out );
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2009-05-01 14:05:46 -07:00
* Switches output format , updates format strings
2008-05-30 11:40:08 +00:00
*
2009-05-01 14:05:46 -07:00
* @ param string $format Format to use , including 'js' for JavaScript - enhanced HTML , 'html' for
* straight HTML output , or 'text' for unformatted text .
2008-05-30 11:40:08 +00:00
* @ access protected
*/
2009-05-01 14:05:46 -07:00
function output ( $format = null , $strings = array ()) {
2008-05-30 11:40:08 +00:00
$_this = Debugger :: getInstance ();
$data = null ;
2009-05-01 14:05:46 -07:00
if ( is_null ( $format )) {
return $_this -> _outputFormat ;
}
if ( ! empty ( $strings )) {
if ( isset ( $_this -> _templates [ $format ])) {
if ( isset ( $strings [ 'links' ])) {
$_this -> _templates [ $format ][ 'links' ] = array_merge (
$_this -> _templates [ $format ][ 'links' ],
$strings [ 'links' ]
);
unset ( $strings [ 'links' ]);
}
$_this -> _templates [ $format ] = array_merge ( $_this -> _templates [ $format ], $strings );
} else {
$_this -> _templates [ $format ] = $strings ;
}
return $_this -> _templates [ $format ];
}
if ( $format === true && ! empty ( $_this -> _data )) {
$data = $_this -> _data ;
$_this -> _data = array ();
2008-05-30 11:40:08 +00:00
$format = false ;
}
2008-12-07 20:44:00 +00:00
$_this -> _outputFormat = $format ;
2008-05-30 11:40:08 +00:00
return $data ;
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2009-05-01 14:05:46 -07:00
* Renders error messages
2008-05-30 11:40:08 +00:00
*
2009-05-01 14:05:46 -07:00
* @ param array $data Data about the current error
2008-05-30 11:40:08 +00:00
* @ access private
*/
2009-05-01 14:05:46 -07:00
function _output ( $data = array ()) {
$defaults = array (
'level' => 0 ,
'error' => 0 ,
'code' => 0 ,
'helpID' => null ,
'description' => '' ,
'file' => '' ,
'line' => 0 ,
'context' => array ()
);
$data += $defaults ;
2008-10-12 03:52:24 +00:00
$files = $this -> trace ( array ( 'start' => 2 , 'format' => 'points' ));
2009-05-01 14:05:46 -07:00
$code = $this -> excerpt ( $files [ 0 ][ 'file' ], $files [ 0 ][ 'line' ] - 1 , 1 );
2008-10-12 03:52:24 +00:00
$trace = $this -> trace ( array ( 'start' => 2 , 'depth' => '20' ));
2009-05-01 14:05:46 -07:00
$insertOpts = array ( 'before' => '{:' , 'after' => '}' );
2008-05-30 11:40:08 +00:00
$context = array ();
2009-05-01 14:05:46 -07:00
$links = array ();
$info = '' ;
2008-05-30 11:40:08 +00:00
2009-05-01 14:05:46 -07:00
foreach (( array ) $data [ 'context' ] as $var => $value ) {
2008-10-12 03:52:24 +00:00
$context [] = " \$ { $var } \t = \t " . $this -> exportVar ( $value , 1 );
2008-05-30 11:40:08 +00:00
}
2008-12-07 20:44:00 +00:00
switch ( $this -> _outputFormat ) {
2009-05-01 14:05:46 -07:00
case false :
$this -> _data [] = compact ( 'context' , 'trace' ) + $data ;
return ;
case 'log' :
$this -> log ( compact ( 'context' , 'trace' ) + $data );
return ;
}
2008-05-30 11:40:08 +00:00
2009-05-01 14:05:46 -07:00
if ( empty ( $this -> _outputFormat ) || ! isset ( $this -> _templates [ $this -> _outputFormat ])) {
$this -> _outputFormat = 'js' ;
}
$data [ 'id' ] = 'cakeErr' . count ( $this -> errors );
$tpl = array_merge ( $this -> _templates [ 'base' ], $this -> _templates [ $this -> _outputFormat ]);
$insert = array ( 'context' => join ( " \n " , $context ), 'helpPath' => $this -> helpPath ) + $data ;
$detect = array ( 'help' => 'helpID' , 'context' => 'context' );
if ( isset ( $tpl [ 'links' ])) {
foreach ( $tpl [ 'links' ] as $key => $val ) {
if ( isset ( $detect [ $key ]) && empty ( $insert [ $detect [ $key ]])) {
continue ;
2008-05-30 11:40:08 +00:00
}
2009-05-01 14:05:46 -07:00
$links [ $key ] = String :: insert ( $val , $insert , $insertOpts );
}
2008-05-30 11:40:08 +00:00
}
2009-05-01 14:05:46 -07:00
foreach ( array ( 'code' , 'context' , 'trace' ) as $key ) {
if ( empty ( $$key ) || ! isset ( $tpl [ $key ])) {
continue ;
}
if ( is_array ( $$key )) {
$$key = join ( " \n " , $$key );
}
$info .= String :: insert ( $tpl [ $key ], compact ( $key ) + $insert , $insertOpts );
}
$links = join ( ' | ' , $links );
unset ( $data [ 'context' ]);
echo String :: insert ( $tpl [ 'error' ], compact ( 'links' , 'info' ) + $data , $insertOpts );
2008-05-30 11:40:08 +00:00
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2008-10-31 20:17:26 +00:00
* Verifies that the application ' s salt value has been changed from the default value .
2008-05-30 11:40:08 +00:00
*
* @ access public
2008-09-25 16:49:56 +00:00
* @ static
2008-05-30 11:40:08 +00:00
*/
function checkSessionKey () {
if ( Configure :: read ( 'Security.salt' ) == 'DYhG93b0qyJfIxfs2guVoUubWwvniR2G0FgaC9mi' ) {
trigger_error ( __ ( 'Please change the value of \'Security.salt\' in app/config/core.php to a salt value specific to your application' , true ), E_USER_NOTICE );
}
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
/**
2009-05-01 14:05:46 -07:00
* Invokes the given debugger object as the current error handler , taking over control from the
* previous handler in a stack - like hierarchy .
2008-05-30 11:40:08 +00:00
*
* @ param object $debugger A reference to the Debugger object
* @ access public
2008-09-25 16:49:56 +00:00
* @ static
2008-10-31 20:17:26 +00:00
* @ link http :// book . cakephp . org / view / 460 / Using - the - Debugger - Class
2008-05-30 11:40:08 +00:00
*/
function invoke ( & $debugger ) {
set_error_handler ( array ( & $debugger , 'handleError' ));
}
}
if ( ! defined ( 'DISABLE_DEFAULT_ERROR_HANDLING' )) {
Debugger :: invoke ( Debugger :: getInstance ());
}
2009-05-01 14:05:46 -07:00
2008-05-30 11:40:08 +00:00
?>