2007-07-19 10:40:28 +00:00
< ? php
2008-02-25 02:10:37 +00:00
/**
2008-03-16 22:13:31 +00:00
* Supports debugging and core error handling .
*
* Attaches custom methods to the default
* error handling hooks in PHP . Currently , three levels
* of error are supported :
*
* - Notice
* - Warning
* - Error
*
* Notice level errors are currently unsupported , and will be passed
* directly to the normal PHP error output .
*
* Uncaught exceptions are currently passed to the debug
* reporter as standard PHP errors .
*
* There are four different types of error handler supported by the
* Debug class :
*
* - Friendly
* - Fatal
* - Logger
* - Emailer
*
* Currently , only Friendly , Fatal , and Emailer handlers are implemented .
*
2008-07-11 06:22:50 +00:00
* @ todo make sure these doc comments are synced with tickets in trac
2008-03-16 22:13:31 +00:00
* @ todo port header / footer wrapping code to external reporter class
* @ todo add support for user defined config : Debug :: die_on_notice ( true | false )
* @ todo add appropriate handling for E_NOTICE and E_USER_NOTICE levels
* @ todo better way of figuring out the error context to display in highlighted source
* @ todo implement error logger handler
2008-02-25 02:10:37 +00:00
*
* @ package sapphire
* @ subpackage core
2007-07-19 10:40:28 +00:00
*/
class Debug {
/**
2008-02-25 02:10:37 +00:00
* @ var $custom_smtp_server string Custom mailserver for sending mails .
2007-07-19 10:40:28 +00:00
*/
protected static $custom_smtp_server = '' ;
2008-02-25 02:10:37 +00:00
/**
* @ var $send_errors_to string Email address to send error notifications
*/
2007-07-19 10:40:28 +00:00
protected static $send_errors_to ;
2008-02-25 02:10:37 +00:00
/**
* @ var $send_warnings_to string Email address to send warning notifications
*/
2007-07-19 10:40:28 +00:00
protected static $send_warnings_to ;
/**
* Show the contents of val in a debug - friendly way .
* Debug :: show () is intended to be equivalent to dprintr ()
*/
static function show ( $val , $showHeader = true ) {
if ( ! Director :: isLive ()) {
if ( $showHeader ) {
$caller = Debug :: caller ();
if ( Director :: is_ajax ())
echo " Debug ( $caller[class] $caller[type] $caller[function] () in line $caller[line] of " . basename ( $caller [ 'file' ]) . " ) \n " ;
else
2008-03-16 22:13:31 +00:00
echo " <div style= \" background-color: white; text-align: left; \" > \n <hr> \n <h3>Debug <span style= \" font-size: 65% \" >( $caller[class] $caller[type] $caller[function] () \n <span style= \" font-weight:normal \" >in line</span> $caller[line] \n <span style= \" font-weight:normal \" >of</span> " . basename ( $caller [ 'file' ]) . " )</span> \n </h3> \n " ;
2007-07-19 10:40:28 +00:00
}
echo Debug :: text ( $val );
if ( ! Director :: is_ajax ()) echo " </div> " ;
}
}
2008-02-25 02:10:37 +00:00
/**
* Emails the contents of the output buffer
*/
2007-07-19 10:40:28 +00:00
static function mailBuffer ( $email , $subject ) {
mail ( $email , $subject , ob_get_contents () );
ob_end_clean ();
}
static function endshow ( $val ) {
if ( ! Director :: isLive ()) {
$caller = Debug :: caller ();
echo " <hr> \n <h3>Debug \n <span style= \" font-size: 65% \" >( $caller[class] $caller[type] $caller[function] () \n <span style= \" font-weight:normal \" >in line</span> $caller[line] \n <span style= \" font-weight:normal \" >of</span> " . basename ( $caller [ 'file' ]) . " )</span> \n </h3> \n " ;
echo Debug :: text ( $val );
die ();
}
}
2008-03-16 22:13:31 +00:00
static function dump ( $val ) {
echo '<pre style="background-color:#ccc;padding:5px;">' ;
print_r ( $val );
echo '</pre>' ;
}
2007-07-19 10:40:28 +00:00
static function text ( $val ) {
2008-02-25 02:10:37 +00:00
if ( is_object ( $val )) {
if ( method_exists ( $val , 'hasMethod' )) {
$hasDebugMethod = $val -> hasMethod ( 'debug' );
2008-02-25 01:06:39 +00:00
} else {
2008-02-25 02:10:37 +00:00
$hasDebugMethod = method_exists ( $val , 'debug' );
}
if ( $hasDebugMethod ) {
return $val -> debug ();
}
}
if ( is_array ( $val )) {
$result = " <ul> \n " ;
foreach ( $val as $k => $v ) {
$result .= " <li> $k = " . Debug :: text ( $v ) . " </li> \n " ;
2007-07-19 10:40:28 +00:00
}
2008-02-25 02:10:37 +00:00
$val = $result . " </ul> \n " ;
2007-07-19 10:40:28 +00:00
2008-02-25 02:10:37 +00:00
} else if ( is_object ( $val )) {
$val = var_export ( $val , true );
} else {
if ( true || ! Director :: is_ajax ()) {
$val = " <pre style= \" font-family: Courier new \" > " . htmlentities ( $val ) . " </pre> \n " ;
}
2007-07-19 10:40:28 +00:00
}
2008-02-25 02:10:37 +00:00
return $val ;
2007-07-19 10:40:28 +00:00
}
2007-09-14 03:19:34 +00:00
2007-07-19 10:40:28 +00:00
/**
* Show a debugging message
*/
static function message ( $message , $showHeader = true ) {
if ( ! Director :: isLive ()) {
$caller = Debug :: caller ();
$file = basename ( $caller [ 'file' ]);
echo " <p style= \" background-color: white; color: black; width: 95%; margin: 0.5em; padding: 0.3em; border: 1px #CCC solid \" > \n " ;
if ( $showHeader ) echo " <b>Debug (line $caller[line] of $file ):</b> \n " ;
echo Convert :: raw2xml ( trim ( $message )) . " </p> \n " ;
}
}
2007-09-14 03:19:34 +00:00
2007-07-19 10:40:28 +00:00
/**
* Load an error handler
2008-02-25 02:10:37 +00:00
*
* @ todo why does this delegate to loadFatalErrorHandler ?
2007-07-19 10:40:28 +00:00
*/
static function loadErrorHandlers () {
Debug :: loadFatalErrorHandler ();
}
2007-09-14 03:19:34 +00:00
2008-02-25 02:10:37 +00:00
/**
* @ todo can this be moved into loadErrorHandlers ?
*/
2007-07-19 10:40:28 +00:00
static function loadFatalErrorHandler () {
2008-03-16 22:13:31 +00:00
//set_error_handler('errorHandler', (E_ALL ^ E_NOTICE) ^ E_USER_NOTICE);
set_error_handler ( 'errorHandler' , E_ALL );
2008-02-25 02:10:37 +00:00
set_exception_handler ( 'exceptionHandler' );
2007-07-19 10:40:28 +00:00
}
static function warningHandler ( $errno , $errstr , $errfile , $errline , $errcontext ) {
2008-02-25 02:10:37 +00:00
if ( error_reporting () == 0 ) return ;
2007-07-19 10:40:28 +00:00
if ( self :: $send_warnings_to ) self :: emailError ( self :: $send_warnings_to , $errno , $errstr , $errfile , $errline , $errcontext , " Warning " );
2007-09-14 03:19:34 +00:00
2007-07-19 10:40:28 +00:00
if ( Director :: isDev ()) {
2008-02-25 02:10:37 +00:00
self :: showError ( $errno , $errstr , $errfile , $errline , $errcontext );
2007-07-19 10:40:28 +00:00
}
}
2007-09-14 03:19:34 +00:00
2007-07-19 10:40:28 +00:00
static function fatalHandler ( $errno , $errstr , $errfile , $errline , $errcontext ) {
if ( self :: $send_errors_to ) self :: emailError ( self :: $send_errors_to , $errno , $errstr , $errfile , $errline , $errcontext , " Error " );
if ( Director :: isDev ()) {
Debug :: showError ( $errno , $errstr , $errfile , $errline , $errcontext );
2007-09-14 03:19:34 +00:00
2007-07-19 10:40:28 +00:00
} else {
Debug :: friendlyError ( $errno , $errstr , $errfile , $errline , $errcontext );
}
die ();
}
static function friendlyError ( $errno , $errstr , $errfile , $errline , $errcontext ) {
header ( " HTTP/1.0 500 Internal server error " );
if ( Director :: is_ajax ()) {
2008-02-25 02:10:37 +00:00
echo " There has been an error " ;
2007-09-14 03:19:34 +00:00
2007-07-19 10:40:28 +00:00
} else {
if ( file_exists ( '../assets/error-500.html' )) {
include ( '../assets/error-500.html' );
} else {
2008-02-25 02:10:37 +00:00
echo " <h1>Error</h1><p>The website server has not been able to respond to your request.</p> \n " ;
2007-07-19 10:40:28 +00:00
}
}
}
static function showError ( $errno , $errstr , $errfile , $errline , $errcontext ) {
if ( ! headers_sent ()) header ( " HTTP/1.0 500 Internal server error " );
if ( Director :: is_ajax ()) {
echo " ERROR:Error $errno : $errstr\n At l $errline in $errfile\n " ;
Debug :: backtrace ();
} else {
2008-06-06 05:13:18 +00:00
$reporter = new DebugView ();
Merged revisions 53150,53681,53700,53820,54200,54459 via svnmerge from
svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/branches/roa
........
r53150 | ischommer | 2008-04-22 11:12:43 +1200 (Tue, 22 Apr 2008) | 1 line
FEATURE Added a "test mode" for /db/build which allows mock-DataObject-subclasses which are just built in a test run
........
r53681 | mrickerby | 2008-04-29 15:26:52 +1200 (Tue, 29 Apr 2008) | 1 line
adding default wrapping header and footer methods, and configurable reporting to the TestRunner
........
r53700 | mrickerby | 2008-04-29 16:41:57 +1200 (Tue, 29 Apr 2008) | 1 line
FEATURE: adding support for /dev/tests --> DevelopmentAdmin-->tests() --> TestRunner, /dev/tasks --> DevelopmentAdmin-->tasks() --> TaskRunner
........
r53820 | mrickerby | 2008-04-30 19:27:52 +1200 (Wed, 30 Apr 2008) | 1 line
BUGFIX fixing up BuildTask interface and task runner action
........
r54200 | sminnee | 2008-05-09 00:28:44 +1200 (Fri, 09 May 2008) | 1 line
Added TestSession object to help with the testing of forms
........
r54459 | sminnee | 2008-05-13 17:28:25 +1200 (Tue, 13 May 2008) | 1 line
Added a basic menu of options to /dev
........
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@54456 467b73ca-7a2a-4603-9d3b-597d59a354a9
2008-05-13 05:57:09 +00:00
$reporter -> writeHeader ();
2008-03-16 22:13:31 +00:00
echo '<div class="info">' ;
echo " <h1> " . strip_tags ( $errstr ) . " </h1> " ;
echo " <h3> { $_SERVER [ 'REQUEST_METHOD' ] } { $_SERVER [ 'REQUEST_URI' ] } </h3> " ;
echo " <p>Line <strong> $errline </strong> in <strong> $errfile </strong></p> " ;
echo '</div>' ;
echo '<div class="trace"><h3>Source</h3>' ;
$lines = file ( $errfile );
$offset = $errline - 10 ;
$lines = array_slice ( $lines , $offset , 16 );
echo '<pre>' ;
$offset ++ ;
foreach ( $lines as $line ) {
$line = htmlentities ( $line );
if ( $offset == $errline ) {
echo " <span> $offset </span> <span class= \" error \" > $line </span> " ;
} else {
echo " <span> $offset </span> $line " ;
}
$offset ++ ;
}
echo '</pre><h3>Trace</h3>' ;
2007-07-19 10:40:28 +00:00
Debug :: backtrace ();
2008-03-16 22:13:31 +00:00
echo '</div>' ;
Merged revisions 53150,53681,53700,53820,54200,54459 via svnmerge from
svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/branches/roa
........
r53150 | ischommer | 2008-04-22 11:12:43 +1200 (Tue, 22 Apr 2008) | 1 line
FEATURE Added a "test mode" for /db/build which allows mock-DataObject-subclasses which are just built in a test run
........
r53681 | mrickerby | 2008-04-29 15:26:52 +1200 (Tue, 29 Apr 2008) | 1 line
adding default wrapping header and footer methods, and configurable reporting to the TestRunner
........
r53700 | mrickerby | 2008-04-29 16:41:57 +1200 (Tue, 29 Apr 2008) | 1 line
FEATURE: adding support for /dev/tests --> DevelopmentAdmin-->tests() --> TestRunner, /dev/tasks --> DevelopmentAdmin-->tasks() --> TaskRunner
........
r53820 | mrickerby | 2008-04-30 19:27:52 +1200 (Wed, 30 Apr 2008) | 1 line
BUGFIX fixing up BuildTask interface and task runner action
........
r54200 | sminnee | 2008-05-09 00:28:44 +1200 (Fri, 09 May 2008) | 1 line
Added TestSession object to help with the testing of forms
........
r54459 | sminnee | 2008-05-13 17:28:25 +1200 (Tue, 13 May 2008) | 1 line
Added a basic menu of options to /dev
........
git-svn-id: svn://svn.silverstripe.com/silverstripe/open/modules/sapphire/trunk@54456 467b73ca-7a2a-4603-9d3b-597d59a354a9
2008-05-13 05:57:09 +00:00
$reporter -> writeFooter ();
2008-03-16 22:13:31 +00:00
die ();
2007-07-19 10:40:28 +00:00
}
}
2007-09-14 03:19:34 +00:00
2007-07-19 10:40:28 +00:00
static function emailError ( $emailAddress , $errno , $errstr , $errfile , $errline , $errcontext , $errorType = " Error " ) {
if ( strtolower ( $errorType ) == 'warning' ) {
$colour = " orange " ;
} else {
$colour = " red " ;
}
2007-09-14 03:19:34 +00:00
2007-07-19 10:40:28 +00:00
$data = " <div style= \" border: 5px $colour solid \" > \n " ;
$data .= " <p style= \" color: white; background-color: $colour ; margin: 0 \" > $errorType : $errstr <br /> At line $errline in $errfile\n <br /> \n <br /> \n </p> \n " ;
$data .= Debug :: backtrace ( true );
$data .= " </div> \n " ;
// override smtp-server if needed
if ( self :: $custom_smtp_server ) {
ini_set ( " SMTP " , self :: $custom_smtp_server );
}
2008-02-25 02:10:37 +00:00
$relfile = Director :: makeRelative ( $errfile );
if ( $relfile [ 0 ] == '/' ) $relfile = substr ( $relfile , 1 );
mail ( $emailAddress , " $errorType at $relfile line $errline (http:// $_SERVER[HTTP_HOST] $_SERVER[REQUEST_URI] ) " , $data , " Content-type: text/html \n From: errors@silverstripe.com " );
2007-07-19 10:40:28 +00:00
}
/**
* @ param string $server IP - Address or domain
*/
static function set_custom_smtp_server ( $server ) {
self :: $custom_smtp_server = $server ;
}
/**
* @ return string
*/
static function get_custom_smtp_server () {
return self :: $custom_smtp_server ;
}
/**
* Send errors to the given email address .
* Can be used like so :
* if ( Director :: isLive ()) Debug :: send_errors_to ( " sam@silverstripe.com " );
* @ param emailAddress The email address to send them to
* @ param sendWarnings Set to true to send warnings as well as errors .
*/
static function send_errors_to ( $emailAddress , $sendWarnings = true ) {
self :: $send_errors_to = $emailAddress ;
self :: $send_warnings_to = $sendWarnings ? $emailAddress : null ;
}
/**
* @ return string
*/
static function get_send_errors_to () {
return self :: $send_errors_to ;
}
/**
* @ return string
*/
static function get_send_warnings_to () {
return self :: $send_warnings_to ;
}
/**
* Deprecated . Send live errors and warnings to the given address .
2008-02-25 02:10:37 +00:00
* @ deprecated Use send_errors_to () instead .
2007-07-19 10:40:28 +00:00
*/
static function sendLiveErrorsTo ( $emailAddress ) {
2008-02-25 02:10:37 +00:00
user_error ( 'Debug::sendLiveErrorsTo() is deprecated. Use Debug::send_errors_to() instead.' , E_USER_NOTICE );
2007-07-19 10:40:28 +00:00
if ( ! Director :: isDev ()) self :: send_errors_to ( $emailAddress , true );
}
static function caller () {
$bt = debug_backtrace ();
$caller = $bt [ 2 ];
$caller [ 'line' ] = $bt [ 1 ][ 'line' ];
$caller [ 'file' ] = $bt [ 1 ][ 'file' ];
2007-08-27 05:13:43 +00:00
if ( ! isset ( $caller [ 'class' ])) $caller [ 'class' ] = '' ;
if ( ! isset ( $caller [ 'type' ])) $caller [ 'type' ] = '' ;
2007-07-19 10:40:28 +00:00
return $caller ;
}
static function backtrace ( $returnVal = false , $ignoreAjax = false ) {
$bt = debug_backtrace ();
// Ingore functions that are plumbing of the error handler
2008-02-25 02:10:37 +00:00
$ignoredFunctions = array ( 'Debug::emailError' , 'Debug::warningHandler' , 'Debug::fatalHandler' , 'errorHandler' , 'Debug::showError' , 'Debug::backtrace' , 'exceptionHandler' );
2007-07-19 10:40:28 +00:00
while ( $bt && in_array ( self :: full_func_name ( $bt [ 0 ]), $ignoredFunctions ) ) {
array_shift ( $bt );
}
2008-02-25 02:10:37 +00:00
2008-03-16 22:13:31 +00:00
$result = " <ul> " ;
2007-07-19 10:40:28 +00:00
foreach ( $bt as $item ) {
if ( Director :: is_ajax () && ! $ignoreAjax ) {
$result .= self :: full_func_name ( $item , true ) . " \n " ;
$result .= " line $item[line] of " . basename ( $item [ 'file' ]) . " \n \n " ;
} else {
2008-03-16 22:13:31 +00:00
if ( $item [ 'function' ] == 'user_error' ) {
$name = $item [ 'args' ][ 0 ];
} else {
$name = self :: full_func_name ( $item , true );
}
$result .= " <li><b> " . $name . " </b> \n <br /> \n " ;
$result .= isset ( $item [ 'line' ]) ? " Line $item[line] of " : '' ;
2007-07-19 10:40:28 +00:00
$result .= isset ( $item [ 'file' ]) ? basename ( $item [ 'file' ]) : '' ;
2008-03-16 22:13:31 +00:00
$result .= " </li> \n " ;
2007-07-19 10:40:28 +00:00
}
}
2008-03-16 22:13:31 +00:00
$result .= " </ul> " ;
2008-02-25 02:10:37 +00:00
if ( $returnVal ) {
return $result ;
} else {
echo $result ;
}
2007-07-19 10:40:28 +00:00
}
/**
* Return the full function name . If showArgs is set to true , a string representation of the arguments will be shown
*/
static function full_func_name ( $item , $showArgs = false ) {
$funcName = '' ;
if ( isset ( $item [ 'class' ])) $funcName .= $item [ 'class' ];
if ( isset ( $item [ 'type' ])) $funcName .= $item [ 'type' ];
if ( isset ( $item [ 'function' ])) $funcName .= $item [ 'function' ];
if ( $showArgs && isset ( $item [ 'args' ])) {
2008-02-25 02:10:37 +00:00
$args = array ();
foreach ( $item [ 'args' ] as $arg ) {
if ( ! is_object ( $arg ) || method_exists ( $arg , '__toString' )) {
$args [] = ( string ) $arg ;
} else {
$args [] = get_class ( $arg );
}
}
$funcName .= " ( " . implode ( " , " , $args ) . " ) " ;
2007-07-19 10:40:28 +00:00
}
return $funcName ;
}
2007-11-07 23:46:00 +00:00
/**
* Check if the user has permissions to run URL debug tools ,
* else redirect them to log in .
*/
static function require_developer_login () {
if ( Director :: isDev ()) {
return ;
}
if ( isset ( $_SESSION [ 'loggedInAs' ])) {
// We have to do some raw SQL here, because this method is called in Object::defineMethods().
// This means we have to be careful about what objects we create, as we don't want Object::defineMethods()
// being called again.
// This basically calls Permission::checkMember($_SESSION['loggedInAs'], 'ADMIN');
$memberID = $_SESSION [ 'loggedInAs' ];
$groups = DB :: query ( " SELECT GroupID from Group_Members WHERE MemberID= " . $memberID );
$groupCSV = implode ( $groups -> column (), ',' );
$permission = DB :: query ( "
SELECT ID
FROM Permission
WHERE (
Code = 'ADMIN'
AND Type = " . Permission::GRANT_PERMISSION . "
AND GroupID IN ( $groupCSV )
)
" )->value();
if ( $permission ) {
return ;
}
}
// This basically does the same as
// Security::permissionFailure(null, "You need to login with developer access to make use of debugging tools.");
// We have to do this because of how early this method is called in execution.
$_SESSION [ 'Security' ][ 'Message' ][ 'message' ] = " You need to login with developer access to make use of debugging tools. " ;
$_SESSION [ 'Security' ][ 'Message' ][ 'type' ] = 'warning' ;
$_SESSION [ 'BackURL' ] = $_SERVER [ 'REQUEST_URI' ];
header ( " HTTP/1.1 302 Found " );
header ( " Location: " . Director :: baseURL () . " Security/login " );
die ();
}
2007-07-19 10:40:28 +00:00
}
2007-11-07 23:46:00 +00:00
2008-02-25 02:10:37 +00:00
function exceptionHandler ( $exception ) {
$errno = E_USER_ERROR ;
$type = get_class ( $exception );
$message = " Uncaught " . $type . " : " . $exception -> getMessage ();
$file = $exception -> getFile ();
$line = $exception -> getLine ();
$context = $exception -> getTrace ();
Debug :: fatalHandler ( $errno , $message , $file , $line , $context );
}
2007-07-19 10:40:28 +00:00
function errorHandler ( $errno , $errstr , $errfile , $errline , $errcontext ) {
switch ( $errno ) {
case E_ERROR :
case E_CORE_ERROR :
case E_USER_ERROR :
Debug :: fatalHandler ( $errno , $errstr , $errfile , $errline , $errcontext );
break ;
2008-03-16 22:13:31 +00:00
case E_NOTICE :
2007-07-19 10:40:28 +00:00
case E_WARNING :
case E_CORE_WARNING :
case E_USER_WARNING :
Debug :: warningHandler ( $errno , $errstr , $errfile , $errline , $errcontext );
break ;
2008-03-16 22:13:31 +00:00
2007-07-19 10:40:28 +00:00
}
}
2008-03-16 22:13:31 +00:00
2007-11-07 23:46:00 +00:00
?>