123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365 |
- <?php
- /**
- * error.class.php
- *
- * This contains the custom error handler for SquirrelMail.
- *
- * @copyright 2005-2025 The SquirrelMail Project Team
- * @license http://opensource.org/licenses/gpl-license.php GNU Public License
- * @version $Id$
- * @package squirrelmail
- */
- /** Used defines */
- define('SQM_NOTICE',0);
- define('SQM_WARNING',1);
- define('SQM_ERROR',2);
- define('SQM_STRICT',3);
- // php5 E_STRICT constant (compatibility with php4)
- if (! defined('E_STRICT')) define('E_STRICT',2048);
- // Set docref_root (fixes URLs that link to php manual)
- if (ini_get('docref_root')=='') ini_set('docref_root','http://www.php.net/');
- /**
- * Error Handler class
- *
- * This class contains a custom error handler in order to display
- * user notices/warnings/errors and php notices and warnings in a template
- *
- * @author Marc Groot Koerkamp
- * @package squirrelmail
- */
- class ErrorHandler {
- /**
- * Constructor (PHP5 style, required in some future version of PHP)
- * @param object $oTemplate Template object
- * @param string $sTemplateFile Template containing the error template
- * @since 1.5.1
- */
- function __construct(&$oTemplate, $sTemplateFile) {
- # echo 'init error handler...';
- $this->TemplateName = $sTemplateFile;
- $this->Template =& $oTemplate;
- $this->aErrors = array();
- $this->header_sent = false;
- $this->delayed_errors = false;
- $this->Template->assign('delayed_errors', $this->delayed_errors);
- }
- /**
- * Constructor (PHP4 style, kept for compatibility reasons)
- * @param object $oTemplate Template object
- * @param string $sTemplateFile Template containing the error template
- * @since 1.5.1
- */
- function ErrorHandler(&$oTemplate, $sTemplateFile) {
- self::__construct($oTemplate, $sTemplateFile);
- }
- /**
- * 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=NULL) {
- $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);
- }
|