Commit 33543cbd by Carsten Brandt

improved error handler output on console app

parent d4ad533d
...@@ -217,6 +217,9 @@ Yii Framework 2 Change Log ...@@ -217,6 +217,9 @@ Yii Framework 2 Change Log
- Chg #2281: Renamed `ActiveRecord::create()` to `populateRecord()` and changed signature. This method will not call instantiate() anymore (cebe) - Chg #2281: Renamed `ActiveRecord::create()` to `populateRecord()` and changed signature. This method will not call instantiate() anymore (cebe)
- Chg #2405: The CSS class of `MaskedInput` now defaults to `form-control` (qiangxue) - Chg #2405: The CSS class of `MaskedInput` now defaults to `form-control` (qiangxue)
- Chg #2426: Changed URL creation method signatures to be consistent (samdark) - Chg #2426: Changed URL creation method signatures to be consistent (samdark)
- Chg #2516: Moved error handling from application to ErrorHandler class and fixed problems with HTTP Exception response code (cebe)
- `Yii::$app->exception` has now moved to `Yii::$app->errorHandler->exception`
- `yii\base\ErrorHandler` was split into `yii\web\ErrorHandler` and `yii\console\ErrorHandler`
- Chg #2544: Changed `DetailView`'s `name:format:label` to `attribute:format:label` to match `GridView` (samdark) - Chg #2544: Changed `DetailView`'s `name:format:label` to `attribute:format:label` to match `GridView` (samdark)
- Chg #2603: `yii\base\ErrorException` now extends `\ErrorException` (samdark) - Chg #2603: `yii\base\ErrorException` now extends `\ErrorException` (samdark)
- Chg #2629: `Module::controllerPath` is now read only, and all controller classes must be namespaced under `Module::controllerNamespace`. (qiangxue) - Chg #2629: `Module::controllerPath` is now read only, and all controller classes must be namespaced under `Module::controllerNamespace`. (qiangxue)
......
...@@ -29,10 +29,7 @@ class ErrorException extends \ErrorException ...@@ -29,10 +29,7 @@ class ErrorException extends \ErrorException
*/ */
public function __construct($message = '', $code = 0, $severity = 1, $filename = __FILE__, $lineno = __LINE__, \Exception $previous = null) public function __construct($message = '', $code = 0, $severity = 1, $filename = __FILE__, $lineno = __LINE__, \Exception $previous = null)
{ {
parent::__construct($message, $code, $previous); parent::__construct($message, $code, $severity, $filename, $lineno, $previous);
$this->severity = $severity;
$this->file = $filename;
$this->line = $lineno;
if (function_exists('xdebug_get_function_stack')) { if (function_exists('xdebug_get_function_stack')) {
$trace = array_slice(array_reverse(xdebug_get_function_stack()), 3, -1); $trace = array_slice(array_reverse(xdebug_get_function_stack()), 3, -1);
......
...@@ -53,10 +53,9 @@ class ErrorHandler extends Component ...@@ -53,10 +53,9 @@ class ErrorHandler extends Component
*/ */
public function register() public function register()
{ {
// TODO care for errorReporting set to deprecated or not ini_set('display_errors', false);
ini_set('display_errors', 0);
set_exception_handler([$this, 'handleException']); set_exception_handler([$this, 'handleException']);
set_error_handler([$this, 'handleError']); set_error_handler([$this, 'handleError']); // TODO care for errorReporting set to deprecated or not
if ($this->memoryReserveSize > 0) { if ($this->memoryReserveSize > 0) {
$this->_memoryReserve = str_repeat('x', $this->memoryReserveSize); $this->_memoryReserve = str_repeat('x', $this->memoryReserveSize);
} }
...@@ -75,16 +74,19 @@ class ErrorHandler extends Component ...@@ -75,16 +74,19 @@ class ErrorHandler extends Component
$this->exception = $exception; $this->exception = $exception;
// disable error capturing to avoid recursive errors while handling exceptions // disable error capturing to avoid recursive errors while handling exceptions
restore_error_handler(); // TODO maybe not needed restore_error_handler();
restore_exception_handler(); restore_exception_handler();
try { try {
$this->logException($exception); $this->logException($exception);
// TODO set HTTP header when HTTP exception
if ($this->discardExistingOutput) { if ($this->discardExistingOutput) {
$this->clearOutput(); $this->clearOutput();
} }
echo $this->renderException($exception); if (PHP_SAPI === 'cli') {
if (PHP_SAPI === 'cli' && !YII_ENV_TEST) { // TODO rethink condition Console::stderr($this->renderException($exception));
} else {
echo $this->renderException($exception);
}
if (!YII_ENV_TEST) {
exit(1); exit(1);
} }
} catch (\Exception $e) { } catch (\Exception $e) {
...@@ -119,7 +121,7 @@ class ErrorHandler extends Component ...@@ -119,7 +121,7 @@ class ErrorHandler extends Component
*/ */
public function handleError($code, $message, $file, $line) public function handleError($code, $message, $file, $line)
{ {
if (error_reporting() & $code) { if (error_reporting() & $code) { // TODO care for errorReporting set to deprecated or not
// load ErrorException manually here because autoloading them will not work // load ErrorException manually here because autoloading them will not work
// when error occurs while autoloading a class // when error occurs while autoloading a class
if (!class_exists('\\yii\\base\\ErrorException', false)) { if (!class_exists('\\yii\\base\\ErrorException', false)) {
...@@ -160,12 +162,17 @@ class ErrorHandler extends Component ...@@ -160,12 +162,17 @@ class ErrorHandler extends Component
$exception = new ErrorException($error['message'], $error['type'], $error['type'], $error['file'], $error['line']); $exception = new ErrorException($error['message'], $error['type'], $error['type'], $error['file'], $error['line']);
$this->exception = $exception; $this->exception = $exception;
// use error_log because it's too late to use Yii log // use error_log because it's too late to use Yii log
error_log($exception); // also do not log when on CLI SAPI because message will be sent to STDERR which has already been done by PHP
PHP_SAPI === 'cli' or error_log($exception);
if ($this->discardExistingOutput) { if ($this->discardExistingOutput) {
$this->clearOutput(); $this->clearOutput();
} }
echo $this->renderException($exception); if (PHP_SAPI === 'cli') {
Console::stderr($this->renderException($exception));
} else {
echo $this->renderException($exception);
}
exit(1); exit(1);
} }
} }
...@@ -178,17 +185,45 @@ class ErrorHandler extends Component ...@@ -178,17 +185,45 @@ class ErrorHandler extends Component
protected function renderException($exception) protected function renderException($exception)
{ {
if ($exception instanceof Exception && ($exception instanceof UserException || !YII_DEBUG)) { if ($exception instanceof Exception && ($exception instanceof UserException || !YII_DEBUG)) {
$message = $exception->getName() . ': ' . $exception->getMessage(); $message = $this->formatMessage($exception->getName() . ': ') . $exception->getMessage();
if (Yii::$app->controller instanceof \yii\console\Controller) { } elseif (YII_DEBUG) {
$message = Yii::$app->controller->ansiFormat($message, Console::FG_RED); if ($exception instanceof Exception) {
$message = $this->formatMessage("Exception ({$exception->getName()})");
} elseif ($exception instanceof ErrorException) {
$message = $this->formatMessage($exception->getName());
} else {
$message = $this->formatMessage('Exception');
} }
$message .= $this->formatMessage(" '" . get_class($exception) . "'", [Console::BOLD, Console::FG_BLUE])
. " with message " . $this->formatMessage("'{$exception->getMessage()}'", [Console::BOLD]) //. "\n"
. "\n\nin " . dirname($exception->getFile()) . DIRECTORY_SEPARATOR . $this->formatMessage(basename($exception->getFile()), [Console::BOLD])
. ':' . $this->formatMessage($exception->getLine(), [Console::BOLD, Console::FG_YELLOW]) . "\n\n"
. $this->formatMessage("Stack trace:\n", [Console::BOLD]) . $exception->getTraceAsString();
} else { } else {
$message = YII_DEBUG ? (string) $exception : 'Error: ' . $exception->getMessage(); // TODO improve output $message = $this->formatMessage('Error: ') . $exception->getMessage();
} }
return $message . "\n"; return $message . "\n";
} }
/** /**
* Colorizes a message for console output.
* @param string $message the message to colorize.
* @param array $format the message format.
* @return string the colorized message.
* @see Console::ansiFormat() for details on how to specify the message format.
*/
protected function formatMessage($message, $format = [Console::FG_RED, Console::BOLD])
{
$stream = (PHP_SAPI === 'cli') ? STDERR : STDOUT;
// try controller first to allow check for --color switch
if (Yii::$app->controller instanceof \yii\console\Controller && Yii::$app->controller->isColorEnabled($stream)
|| Yii::$app instanceof \yii\console\Application && Console::streamSupportsAnsiColors($stream)) {
$message = Console::ansiFormat($message, $format);
}
return $message;
}
/**
* Logs the given exception * Logs the given exception
* @param \Exception $exception the exception to be logged * @param \Exception $exception the exception to be logged
*/ */
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment