X-Git-Url: http://git.cyclocoop.org/%7B%24admin_url%7Dmembres/cotisations/gestion/rappel_supprimer.php?a=blobdiff_plain;f=includes%2FException.php;h=015298d4302602ca1857eb2d26939107ba92346c;hb=3e11266ae5e5a33498abb7f028bbc79b5ca93515;hp=696454d052a9312347a9f4e7be3032a010adaec2;hpb=bcb9f9e1c0dda36bfe2096e92de0ad77e21b38f8;p=lhc%2Fweb%2Fwiklou.git diff --git a/includes/Exception.php b/includes/Exception.php index 696454d052..015298d430 100644 --- a/includes/Exception.php +++ b/includes/Exception.php @@ -1,117 +1,294 @@ useMessageCache() && + !empty( $GLOBALS['wgFullyInitialised'] ) && + !empty( $GLOBALS['wgOut'] ) && + !$GLOBALS['wgOut']->isArticleRelated() && + !empty( $GLOBALS['wgTitle'] ); } + /** + * Can the extension use wfMsg() to get i18n messages ? + * @return bool + */ function useMessageCache() { global $wgLang; - return is_object( $wgLang ); + + foreach ( $this->getTrace() as $frame ) { + if ( isset( $frame['class'] ) && $frame['class'] === 'LocalisationCache' ) { + return false; + } + } + + return $wgLang instanceof Language; + } + + /** + * Run hook to allow extensions to modify the text of the exception + * + * @param $name String: class name of the exception + * @param $args Array: arguments to pass to the callback functions + * @return Mixed: string to output or null if any hook has been called + */ + function runHooks( $name, $args = array() ) { + global $wgExceptionHooks; + + if ( !isset( $wgExceptionHooks ) || !is_array( $wgExceptionHooks ) ) { + return; // Just silently ignore + } + + if ( !array_key_exists( $name, $wgExceptionHooks ) || !is_array( $wgExceptionHooks[ $name ] ) ) { + return; + } + + $hooks = $wgExceptionHooks[ $name ]; + $callargs = array_merge( array( $this ), $args ); + + foreach ( $hooks as $hook ) { + if ( is_string( $hook ) || ( is_array( $hook ) && count( $hook ) >= 2 && is_string( $hook[0] ) ) ) { // 'function' or array( 'class', hook' ) + $result = call_user_func_array( $hook, $callargs ); + } else { + $result = null; + } + + if ( is_string( $result ) ) + return $result; + } } + /** + * Get a message from i18n + * + * @param $key String: message name + * @param $fallback String: default message if the message cache can't be + * called by the exception + * The function also has other parameters that are arguments for the message + * @return String message with arguments replaced + */ function msg( $key, $fallback /*[, params...] */ ) { $args = array_slice( func_get_args(), 2 ); + if ( $this->useMessageCache() ) { - return wfMsgReal( $key, $args ); + return wfMsgNoTrans( $key, $args ); } else { return wfMsgReplaceArgs( $fallback, $args ); } } - + + /** + * If $wgShowExceptionDetails is true, return a HTML message with a + * backtrace to the error, otherwise show a message to ask to set it to true + * to show that information. + * + * @return String html to output + */ function getHTML() { - return '

' . htmlspecialchars( $this->getMessage() ) . - '

Backtrace:

' . nl2br( htmlspecialchars( $this->getTraceAsString() ) ) . - "

\n"; + global $wgShowExceptionDetails; + + if ( $wgShowExceptionDetails ) { + return '

' . nl2br( htmlspecialchars( $this->getMessage() ) ) . + '

Backtrace:

' . nl2br( htmlspecialchars( $this->getTraceAsString() ) ) . + "

\n"; + } else { + return "

Set \$wgShowExceptionDetails = true; " . + "at the bottom of LocalSettings.php to show detailed " . + "debugging information.

"; + } } + /** + * If $wgShowExceptionDetails is true, return a text message with a + * backtrace to the error. + */ function getText() { - return $this->getMessage() . - "\nBacktrace:\n" . $this->getTraceAsString() . "\n"; - } - - function getPageTitle() { - if ( $this->useMessageCache() ) { - return wfMsg( 'internalerror' ); + global $wgShowExceptionDetails; + + if ( $wgShowExceptionDetails ) { + return $this->getMessage() . + "\nBacktrace:\n" . $this->getTraceAsString() . "\n"; } else { - global $wgSitename; - return "$wgSitename error"; + return "Set \$wgShowExceptionDetails = true; " . + "in LocalSettings.php to show detailed debugging information.\n"; } } - + + /* Return titles of this error page */ + function getPageTitle() { + global $wgSitename; + return $this->msg( 'internalerror', "$wgSitename error" ); + } + + /** + * Return the requested URL and point to file and line number from which the + * exception occured + * + * @return String + */ function getLogMessage() { + global $wgRequest; + $file = $this->getFile(); $line = $this->getLine(); $message = $this->getMessage(); - return "{$_SERVER['REQUEST_URI']} Exception from line $line of $file: $message"; + + if ( isset( $wgRequest ) ) { + $url = $wgRequest->getRequestURL(); + if ( !$url ) { + $url = '[no URL]'; + } + } else { + $url = '[no req]'; + } + + return "$url Exception from line $line of $file: $message"; } - + + /** Output the exception report using HTML */ function reportHTML() { global $wgOut; + if ( $this->useOutputPage() ) { $wgOut->setPageTitle( $this->getPageTitle() ); - $wgOut->setRobotpolicy( "noindex,nofollow" ); + $wgOut->setRobotPolicy( "noindex,nofollow" ); $wgOut->setArticleRelated( false ); $wgOut->enableClientCache( false ); $wgOut->redirect( '' ); $wgOut->clearHTML(); - $wgOut->addHTML( $this->getHTML() ); + + $hookResult = $this->runHooks( get_class( $this ) ); + if ( $hookResult ) { + $wgOut->addHTML( $hookResult ); + } else { + $wgOut->addHTML( $this->getHTML() ); + } + $wgOut->output(); } else { - echo $this->htmlHeader(); - echo $this->getHTML(); - echo $this->htmlFooter(); + $hookResult = $this->runHooks( get_class( $this ) . "Raw" ); + if ( $hookResult ) { + die( $hookResult ); + } + + $html = $this->getHTML(); + if ( defined( 'MEDIAWIKI_INSTALL' ) ) { + echo $html; + } else { + wfDie( $html ); + } } } - - function reportText() { - echo $this->getText(); - } + /** + * Output a report about the exception and takes care of formatting. + * It will be either HTML or plain text based on isCommandLine(). + */ function report() { - global $wgCommandLineMode; - if ( $wgCommandLineMode ) { - $this->reportText(); + $log = $this->getLogMessage(); + + if ( $log ) { + wfDebugLog( 'exception', $log ); + } + + if ( self::isCommandLine() ) { + wfPrintError( $this->getText() ); } else { - $log = $this->getLogMessage(); - if ( $log ) { - wfDebugLog( 'exception', $log ); - } $this->reportHTML(); } } + /** + * Send headers and output the beginning of the html page if not using + * $wgOut to output the exception. + * @deprecated since 1.18 call wfDie() if you need to die immediately + */ function htmlHeader() { - global $wgLogo, $wgSitename, $wgOutputEncoding; + global $wgLogo, $wgLang; if ( !headers_sent() ) { header( 'HTTP/1.0 500 Internal Server Error' ); - header( 'Content-type: text/html; charset='.$wgOutputEncoding ); + header( 'Content-type: text/html; charset=UTF-8' ); /* Don't cache error pages! They cause no end of trouble... */ header( 'Cache-control: none' ); header( 'Pragma: nocache' ); } - $title = $this->getPageTitle(); - echo " - - $title - - -

$title

- "; + + $head = Html::element( 'title', null, $this->getPageTitle() ) . "\n"; + $head .= Html::inlineStyle( <<getDir(); + $code = $wgLang->getCode(); + } + + $header = Html::element( 'img', array( + 'src' => $wgLogo, + 'alt' => '' ) ); + + $attribs = array( 'dir' => $dir, 'lang' => $code ); + + return + Html::htmlHeader( $attribs ) . + Html::rawElement( 'head', null, $head ) . "\n". + Html::openElement( 'body' ) . "\n" . + $header . "\n"; + } + + /** + * print the end of the html page if not using $wgOut. + * @deprecated since 1.18 + */ function htmlFooter() { - echo ""; + return Html::closeElement( 'body' ) . Html::closeElement( 'html' ); } + static function isCommandLine() { + return !empty( $GLOBALS['wgCommandLineMode'] ) && !defined( 'MEDIAWIKI_INSTALL' ); + } } /** * Exception class which takes an HTML error message, and does not * produce a backtrace. Replacement for OutputPage::fatalError(). + * @ingroup Exception */ class FatalError extends MWException { function getHTML() { @@ -123,25 +300,155 @@ class FatalError extends MWException { } } +/** + * An error page which can definitely be safely rendered using the OutputPage + * @ingroup Exception + */ class ErrorPageError extends MWException { - public $title, $msg; - + public $title, $msg, $params; + /** * Note: these arguments are keys into wfMsg(), not text! */ - function __construct( $title, $msg ) { + function __construct( $title, $msg, $params = null ) { $this->title = $title; $this->msg = $msg; - parent::__construct( wfMsg( $msg ) ); + $this->params = $params; + + if( $msg instanceof Message ){ + parent::__construct( $msg ); + } else { + parent::__construct( wfMsg( $msg ) ); + } } function report() { global $wgOut; - $wgOut->showErrorPage( $this->title, $this->msg ); + + if ( $wgOut->getTitle() ) { + $wgOut->debug( 'Original title: ' . $wgOut->getTitle()->getPrefixedText() . "\n" ); + } + $wgOut->setPageTitle( wfMsg( $this->title ) ); + $wgOut->setHTMLTitle( wfMsg( 'errorpagetitle' ) ); + $wgOut->setRobotPolicy( 'noindex,nofollow' ); + $wgOut->setArticleRelated( false ); + $wgOut->enableClientCache( false ); + $wgOut->mRedirect = ''; + $wgOut->clearHTML(); + + if( $this->msg instanceof Message ){ + $wgOut->addHTML( $this->msg->parse() ); + } else { + $wgOut->addWikiMsgArray( $this->msg, $this->params ); + } + + $wgOut->returnToMain(); $wgOut->output(); } } +/** + * Show an error when a user tries to do something they do not have the necessary + * permissions for. + */ +class PermissionsError extends ErrorPageError { + public $permission; + + function __construct( $permission ) { + global $wgLang; + + $this->permission = $permission; + + $groups = array_map( + array( 'User', 'makeGroupLinkWiki' ), + User::getGroupsWithPermission( $this->permission ) + ); + + if( $groups ) { + parent::__construct( + 'badaccess', + 'badaccess-groups', + array( + $wgLang->commaList( $groups ), + count( $groups ) + ) + ); + } else { + parent::__construct( + 'badaccess', + 'badaccess-group0' + ); + } + } +} + +/** + * Show an error when the wiki is locked/read-only and the user tries to do + * something that requires write access + */ +class ReadOnlyError extends ErrorPageError { + public function __construct(){ + parent::__construct( + 'readonly', + 'readonlytext', + wfReadOnlyReason() + ); + } +} + +/** + * Show an error when the user hits a rate limit + */ +class ThrottledError extends ErrorPageError { + public function __construct(){ + parent::__construct( + 'actionthrottled', + 'actionthrottledtext' + ); + } + public function report(){ + global $wgOut; + $wgOut->setStatusCode( 503 ); + return parent::report(); + } +} + +/** + * Show an error when the user tries to do something whilst blocked + */ +class UserBlockedError extends ErrorPageError { + public function __construct( Block $block ){ + global $wgLang; + + $blockerUserpage = $block->getBlocker()->getUserPage(); + $link = "[[{$blockerUserpage->getPrefixedText()}|{$blockerUserpage->getText()}]]"; + + $reason = $block->mReason; + if( $reason == '' ) { + $reason = wfMsg( 'blockednoreason' ); + } + + /* $ip returns who *is* being blocked, $intended contains who was meant to be blocked. + * This could be a username, an IP range, or a single IP. */ + $intended = $block->getTarget(); + + parent::__construct( + 'blockedtitle', + $block->mAuto ? 'autoblocketext' : 'blockedtext', + array( + $link, + $reason, + wfGetIP(), + $block->getBlocker()->getName(), + $block->getId(), + $wgLang->formatExpiry( $block->mExpiry ), + $intended, + $wgLang->timeanddate( wfTimestamp( TS_MW, $block->mTimestamp ), true ) + ) + ); + } +} + /** * Install an exception handler for MediaWiki exception types. */ @@ -153,27 +460,65 @@ function wfInstallExceptionHandler() { * Report an exception to the user */ function wfReportException( Exception $e ) { - if ( is_a( $e, 'MWException' ) ) { - try { - $e->report(); - } catch ( Exception $e2 ) { - // Exception occurred from within exception handler - // Show a simpler error message for the original exception, - // don't try to invoke report() - $message = "MediaWiki internal error.\n\n" . - "Original exception: " . $e->__toString() . - "\n\nException caught inside exception handler: " . - $e2->__toString() . "\n"; - - if ( !empty( $GLOBALS['wgCommandLineMode'] ) ) { - echo $message; - } else { - echo nl2br( htmlspecialchars( $message ) ). "\n"; - } - } - } else { - echo $e->__toString(); - } + global $wgShowExceptionDetails; + + $cmdLine = MWException::isCommandLine(); + + if ( $e instanceof MWException ) { + try { + // Try and show the exception prettily, with the normal skin infrastructure + $e->report(); + } catch ( Exception $e2 ) { + // Exception occurred from within exception handler + // Show a simpler error message for the original exception, + // don't try to invoke report() + $message = "MediaWiki internal error.\n\n"; + + if ( $wgShowExceptionDetails ) { + $message .= 'Original exception: ' . $e->__toString() . "\n\n" . + 'Exception caught inside exception handler: ' . $e2->__toString(); + } else { + $message .= "Exception caught inside exception handler.\n\n" . + "Set \$wgShowExceptionDetails = true; at the bottom of LocalSettings.php " . + "to show detailed debugging information."; + } + + $message .= "\n"; + + if ( $cmdLine ) { + wfPrintError( $message ); + } else { + wfDie( nl2br( htmlspecialchars( $message ) ) ) . "\n"; + } + } + } else { + $message = "Unexpected non-MediaWiki exception encountered, of type \"" . get_class( $e ) . "\"\n" . + $e->__toString() . "\n"; + + if ( $wgShowExceptionDetails ) { + $message .= "\n" . $e->getTraceAsString() . "\n"; + } + + if ( $cmdLine ) { + wfPrintError( $message ); + } else { + wfDie( nl2br( htmlspecialchars( $message ) ) ) . "\n"; + } + } +} + +/** + * Print a message, if possible to STDERR. + * Use this in command line mode only (see isCommandLine) + */ +function wfPrintError( $message ) { + # NOTE: STDERR may not be available, especially if php-cgi is used from the command line (bug #15602). + # Try to produce meaningful output anyway. Using echo may corrupt output to STDOUT though. + if ( defined( 'STDERR' ) ) { + fwrite( STDERR, $message ); + } else { + echo( $message ); + } } /** @@ -189,9 +534,10 @@ function wfReportException( Exception $e ) { */ function wfExceptionHandler( $e ) { global $wgFullyInitialised; + wfReportException( $e ); - - // Final cleanup, similar to wfErrorExit() + + // Final cleanup if ( $wgFullyInitialised ) { try { wfLogProfilingData(); // uses $wgRequest, hence the $wgFullyInitialised condition @@ -201,5 +547,3 @@ function wfExceptionHandler( $e ) { // Exit value should be nonzero for the benefit of shell jobs exit( 1 ); } - -?>