TemplateName = $sTemplateFile; $this->Template =& $oTemplate; $this->aErrors = array(); $this->header_sent = false; $this->delayed_errors = false; $this->Template->assign('delayed_errors', $this->delayed_errors); } /** * Sets the error template * @since 1.5.1 */ function SetTemplateFile($sTemplateFile) { $this->TemplateFile = $sTemplateFile; } /** * Sets if the page header is already sent * @since 1.5.1 */ function HeaderSent() { $this->header_sent = true; $this->Template->assign('header_sent', true); } /** * Turn on/off delayed error handling * @since 1.5.2 */ function setDelayedErrors ($val = true) { $this->delayed_errors = $val===true; $this->Template->assign('delayed_errors', $this->delayed_errors); } /** * Store errors generated in a previous script but couldn't be displayed * due to a header redirect. This requires storing of aDelayedErrors in the session * @param array $aDelayedErrors array with errors stored in the $this->aErrors format. * @since 1.5.1 */ function AssignDelayedErrors(&$aDelayedErrors) { $aErrors = array_merge($this->aErrors,$aDelayedErrors); $this->aErrors = $aErrors; $this->Template->assign('aErrors',$this->aErrors); $aDelayedErrors = false; } /** * Custom Error handler (set with set_error_handler() ) * @private * @since 1.5.1 */ function SquirrelMailErrorhandler($iErrNo, $sErrStr, $sErrFile, $iErrLine, $aContext) { $aError = array( 'type' => SQM_NOTICE,// Error type, notice, warning or fatal error; 'category' => NULL, // SquirrelMail error category; 'message' => NULL, // Error display message; 'extra' => NULL, // Key value based array with extra error info; 'link' => NULL, // Link to help location; 'tip' => NULL // User tip. ); $iType = NULL; $aErrorCategory = array(); /** * Get current error reporting level. * * PHP 4.1.2 does not return current error reporting level in ini_get (php 5.1b3 and * 4.3.10 does). Retrieve current error reporting level while setting error reporting * to ini value and reset it to retrieved value. */ $iCurErrLevel = error_reporting(ini_get('error_reporting')); error_reporting($iCurErrLevel); /** * Check error_reporting value before logging error. * Don't log errors that are disabled by @ (error_reporting = 0). Some SquirrelMail scripts * (sq_mb_list_encodings(), ldap function calls in functions/abook_ldap_server.php) * handle errors themselves and @ is used to disable generic php error messages. */ if ($iErrNo & $iCurErrLevel) { /* * The following errors cannot be handled by a user defined error handler: * E_ERROR, E_PARSE, E_CORE_ERROR, E_CORE_WARNING, E_COMPILE_ERROR, E_COMPILE_WARNING */ switch ($iErrNo) { case E_STRICT: $iType = (is_null($iType)) ? SQM_STRICT : $iType; case E_NOTICE: $iType = (is_null($iType)) ? SQM_NOTICE : $iType; case E_WARNING: $iType = (is_null($iType)) ? SQM_WARNING : $iType; $aErrorCategory[] = 'PHP'; $aError['message'] = $sErrStr; $aError['extra'] = array( 'FILE' => $sErrFile, 'LINE' => $iErrLine) ; // what todo with $aContext? break; case E_USER_ERROR: $iType = (is_null($iType)) ? SQM_ERROR : $iType; case E_USER_NOTICE: $iType = (is_null($iType)) ? SQM_NOTICE : $iType; case E_USER_WARNING: $iType = (is_null($iType)) ? SQM_WARNING : $iType; if ($sErrFile == __FILE__) { // Error is triggered in this file and probably by sqm_trigger_error $aErrorTemp = @unserialize($sErrStr); if (!is_array($aErrorTemp)) { $aError['message'] = $sErrStr; $aErrorCategory[] = 'UNDEFINED'; } else { $aError = array_merge($aError,$aErrorTemp); // special error handling below if ($aError['category'] & SQM_ERROR_IMAP) { $aErrorCategory[] = 'IMAP'; // imap related error handling inside } if ($aError['category'] & SQM_ERROR_FS) { $aErrorCategory[] = 'FILESYSTEM'; // filesystem related error handling inside } if ($aError['category'] & SQM_ERROR_SMTP) { $aErrorCategory[] = 'SMTP'; // smtp related error handling inside } if ($aError['category'] & SQM_ERROR_LDAP) { $aErrorCategory[] = 'LDAP'; // ldap related error handling inside } if ($aError['category'] & SQM_ERROR_DB) { $aErrorCategory[] = 'DATABASE'; // db related error handling inside } if ($aError['category'] & SQM_ERROR_PLUGIN) { $aErrorCategory[] = 'PLUGIN'; do_hook('error_handler_plugin', $aError); // plugin related error handling inside } //if ($aError['category'] & SQM_ERROR_X) { // $aErrorCategory[] = 'X'; // place holder for a new category //} } unset($aErrorTemp); } else { $aError['message'] = $sErrStr; $aErrorCategory[] = 'SQM_NOTICE'; } break; default: break; } /** * If delayed error handling is enabled, always record the location * and tag the error is delayed to make debugging easier. */ if (isset($this->Template->values['delayed_errors']) && $this->Template->values['delayed_errors']) { $aErrorCategory[] = 'Delayed'; $aError['extra'] = array( 'FILE' => $sErrFile, 'LINE' => $iErrLine) ; } $aErrorTpl = array( 'type' => $iType, 'category' => $aErrorCategory, 'message' => $aError['message'], 'link' => $aError['link'], 'tip' => $aError['tip'], 'extra' => $aError['extra']); // Add the notice/warning/error to the existing list of notices/warnings $this->aErrors[] = $aErrorTpl; $this->Template->assign('aErrors',$this->aErrors); } // Show the error immediate in case of fatal errors if ($iType == SQM_ERROR) { if (isset($this->Template->values['header_sent']) && !$this->Template->values['header_sent']) { // TODO replace this with template that can be assigned // UPDATE: displayHtmlHeader() no longer sends anything // directly to the browser itself and instead // displays all output through the template file // "protocol_header" as well as calls to the // template's header() method, so perhaps the // above TODO is alleviated?? (however, I don't fully // understand the problem behind the TODO comment myself (Paul)) displayHtmlHeader(_("Error"),'',false); } $this->DisplayErrors(); exit(_("Terminating SquirrelMail due to a fatal error")); } } /** * Force the delayed errors to be stored in the session in case * $this->displayErrors() never gets called, e.g. in compose.php */ function saveDelayedErrors () { if($this->delayed_errors) { // Check for previous delayed errors... sqgetGlobalVar('delayed_errors', $delayed_errors, SQ_SESSION); if (is_array($delayed_errors)) { $this->AssignDelayedErrors($delayed_errors); sqsession_unregister("delayed_errors"); } if (count($this->aErrors) > 0) { sqsession_register($this->aErrors,"delayed_errors"); } } } /** * Display the error array in the error template * @return void * @since 1.5.1 */ function DisplayErrors() { // Check for delayed errors... if (!$this->delayed_errors) { sqgetGlobalVar('delayed_errors', $delayed_errors, SQ_SESSION); if (is_array($delayed_errors)) { $this->AssignDelayedErrors($delayed_errors); sqsession_unregister("delayed_errors"); } } if (isset($this->Template->values['aErrors']) && count($this->Template->values['aErrors']) > 0) { foreach ($this->Template->values['aErrors'] as $err) { if (!in_array($err, $this->aErrors, true)) { $this->aErrors[] = $err; } } $this->Template->assign('aErrors',$this->aErrors); } if (count($this->aErrors) > 0) { if ($this->delayed_errors) { sqsession_register($this->aErrors,"delayed_errors"); } else { $this->Template->display($this->TemplateName); } } } } /** * Custom Error handler for PHP version < 4.3.0 (set with set_error_handler() ) * @author Marc Groot Koerkamp * @since 1.5.1 */ function SquirrelMailErrorhandler($iErrNo, $sErrStr, $sErrFile, $iErrLine, $aContext) { global $oTemplate; static $oErrorHandler; if (!isset($oErrorHandler)) { $oErrorHandler = new ErrorHandler($oTemplate,'error_message.tpl'); } $oErrorHandler->SquirrelMailErrorhandler($iErrNo, $sErrStr, $sErrFile, $iErrLine, $aContext); } /** * Triggers an imap error. Utility function for sqm_trigger_error() * @param string $sErrNo error string defined in errors.php * @param string $sRequest imap request string * @param string $sResponse tagged imap response * @param string $sMessage tagged imap response message * @param array $aExtra optional associative array with extra error info * @return void * @author Marc Groot Koerkamp * @since 1.5.1 */ function sqm_trigger_imap_error($sErrNo,$sRequest,$sResponse, $sMessage, $aExtra=array()) { $aError = array( 'REQUEST' => $sRequest, 'RESPONSE' => $sResponse, 'MESSAGE' => $sMessage); $aError = array_merge($aExtra,$aError); sqm_trigger_error($sErrNo,$aError); } /** * Trigger an error. * @param string $sErrNo error string defined in errors.php * @param array $aExtra optional associative array with extra error info * @return void * @author Marc Groot Koerkamp * @since 1.5.1 */ function sqm_trigger_error($sErrNo,$aExtra=array()) { static $aErrors; if (!isset($aErrors)) { // Include the error definition file. include_once(SM_PATH.'include/errors.php'); } $iPhpErr = E_USER_NOTICE; if (is_array($aErrors) && isset($aErrors[$sErrNo]['level'])) { if (is_array($aExtra) && count($aExtra)) { $aErrors[$sErrNo]['extra'] = $aExtra; } // because trigger_error can only handle a string argument for the error description // we serialize the result. $sErrString = serialize($aErrors[$sErrNo]); $iPhpErr = $aErrors[$sErrNo]['level']; } else { sm_print_r($aErrors); $sErrString = "Error <$sErrNo> does not exist, fix the code or update the errors.php file"; $iPhpErr = E_USER_ERROR; } trigger_error($sErrString, $iPhpErr); }