From 64f3a79aae254b71550a8097880f0b0e09062d24 Mon Sep 17 00:00:00 2001 From: Taylor Otwell Date: Thu, 11 Apr 2013 13:29:54 -0500 Subject: [PATCH] Integrate Whoops into the core of the framework error handling. --- composer.json | 3 +- .../Exception/ExceptionServiceProvider.php | 83 ++++- src/Illuminate/Exception/composer.json | 1 + .../Exception/resources/pretty-page.css | 294 ++++++++++++++++++ .../Exception/resources/pretty-template.php | 195 ++++++++++++ 5 files changed, 571 insertions(+), 5 deletions(-) create mode 100644 src/Illuminate/Exception/resources/pretty-page.css create mode 100644 src/Illuminate/Exception/resources/pretty-template.php diff --git a/composer.json b/composer.json index 7ab9aacf9c7d..4b5291f51e25 100644 --- a/composer.json +++ b/composer.json @@ -13,9 +13,10 @@ "php": ">=5.3.7", "classpreloader/classpreloader": "1.0.*", "ircmaxell/password-compat": "1.0.*", + "filp/whoops": "1.0.0", "monolog/monolog": "1.4.*", "patchwork/utf8": "1.0.*", - "predis/predis": "0.*", + "predis/predis": "0.8.*", "swiftmailer/swiftmailer": "4.3.*", "symfony/browser-kit": "2.3.*", "symfony/console": "2.3.*", diff --git a/src/Illuminate/Exception/ExceptionServiceProvider.php b/src/Illuminate/Exception/ExceptionServiceProvider.php index 9c72835b5121..ec095fdf0e6f 100644 --- a/src/Illuminate/Exception/ExceptionServiceProvider.php +++ b/src/Illuminate/Exception/ExceptionServiceProvider.php @@ -1,7 +1,10 @@ registerExceptionHandler(); + + $this->registerWhoops(); } /** @@ -72,11 +77,11 @@ protected function registerKernelHandlers() */ protected function registerExceptionHandler() { - $app = $this->app; + list($me, $app) = array($this, $this->app); - $app['exception.function'] = function() use ($app) + $app['exception.function'] = function() use ($me, $app) { - return function($exception) use ($app) + return function($exception) use ($me, $app) { $response = $app['exception']->handle($exception); @@ -91,7 +96,7 @@ protected function registerExceptionHandler() } else { - $app['kernel.exception']->handle($exception); + $me->displayException($exception); } }; }; @@ -114,6 +119,76 @@ protected function registerShutdownHandler() }); } + /** + * Register the Whoops error display service. + * + * @return void + */ + protected function registerWhoops() + { + $this->registerWhoopsHandler(); + + $this->app['whoops'] = $this->app->share(function($app) + { + $whoops = new \Whoops\Run; + + $whoops->allowQuit(false); + + return $whoops->pushHandler($app['whoops.handler']); + }); + } + + /** + * Register the Whoops handler for the request. + * + * @return void + */ + protected function registerWhoopsHandler() + { + if ($this->app['request']->ajax() or $this->app->runningInConsole()) + { + $this->app['whoops.handler'] = function() { return new JsonResponseHandler; }; + } + else + { + $this->app['whoops.handler'] = function() + { + with($handler = new PrettyPageHandler)->setResourcesPath(__DIR__.'/resources'); + + return $handler; + }; + } + } + + /** + * Display the given exception. + * + * @param \Exception $exception + * @return void + */ + public function displayException($exception) + { + if ($this->app['config']['app.debug']) + { + return $this->displayWhoopsException($exception); + } + + $this->app['kernel.exception']->handle($exception); + } + + /** + * Display a exception using the Whoops library. + * + * @param \Exception $exception + * @return void + */ + protected function displayWhoopsException($exception) + { + ob_start(); $this->app['whoops']->handleException($exception); + + with(new Response(ob_get_clean(), 500))->send(); + } + /** * Set the given Closure as the exception handler. * diff --git a/src/Illuminate/Exception/composer.json b/src/Illuminate/Exception/composer.json index 5351fdeaaaf6..f70d1320b76b 100644 --- a/src/Illuminate/Exception/composer.json +++ b/src/Illuminate/Exception/composer.json @@ -10,6 +10,7 @@ "require": { "php": ">=5.3.0", "illuminate/support": "4.0.x", + "symfony/http-foundation": "2.3.x", "symfony/http-kernel": "2.3.*" }, "require-dev": { diff --git a/src/Illuminate/Exception/resources/pretty-page.css b/src/Illuminate/Exception/resources/pretty-page.css new file mode 100644 index 000000000000..0d1eca063707 --- /dev/null +++ b/src/Illuminate/Exception/resources/pretty-page.css @@ -0,0 +1,294 @@ +.cf:before, .cf:after {content: " ";display: table;} .cf:after {clear: both;} .cf {*zoom: 1;} +body { + font: 14px helvetica, arial, sans-serif; + color: #2B2B2B; + background-color: #e7e7e7; + padding:0; + margin: 0; + max-height: 100%; +} + a { + text-decoration: none; + } + +.container{ + height: 100%; + width: 100%; + position: fixed; + margin: 0; + padding: 0; +} + +.branding { + position: absolute; + top: 10px; + right: 20px; + color: #777777; + font-size: 10px; + z-index: 100; +} + .branding a { + color: #CD3F3F; + } + +header { + padding: 30px 20px; + color: #555; + background: #ddd; /* 272727 */ + box-sizing: border-box; + border-left: 5px solid #ED3D1A; + background-image: url(); + background-repeat:no-repeat; + background-position:right; +} + .exc-title { + margin: 0; + color: #616161; + /*text-shadow: 0 1px 2px rgba(0, 0, 0, .1);*/ + font-weight:normal; + } + .exc-title-primary { color: #ED591A; } + .exc-message { + font-size: 32px; + margin: 5px 0; + word-wrap: break-word; + } + +.stack-container { + height: 100%; + position: relative; +} + +.details-container { + height: 100%; + overflow: auto; + float: right; + width: 70%; + background: #fff; +} + .details { + padding: 10px 20px; + border-left: 5px solid rgba(0, 0, 0, .2); + } + +.frames-container { + height: 100%; + overflow: auto; + float: left; + width: 30%; +} + .frame { + padding: 14px; + background: #F3F3F3; + border-right: 1px solid rgba(0, 0, 0, .2); + cursor: pointer; + } + .frame.active { + background-color: #ED591A; + color: #F3F3F3; + box-shadow: inset -2px 0 0 rgba(255, 255, 255, .1); + text-shadow: 0 1px 0 rgba(0, 0, 0, .2); + } + + .frame:not(.active):hover { + background: #F0E5DF; + } + + .frame-class, .frame-function { + font-weight: bold; + } + + .frame-class { + color: #ED591A; + } + .active .frame-class { + color: #FFCBB5; + } + + .frame-file { + font-family: consolas, monospace; + word-wrap:break-word; + } + + .frame-line { + font-weight: bold; + color: #A33202; + } + + .active .frame-line { color: #FFCBB5; } + .frame-line:before { + content: ":"; + } + + .frame-code { + padding: 20px; + background: #f0f0f0; + display: none; + border-left: 5px solid #EDA31A; + } + + .frame-code.active { + display: block; + } + + .frame-code .frame-file { + background: #ED591A; + color: #fff; + padding: 10px 10px 5px 10px; +/* + border-top-right-radius: 3px; + border-top-left-radius: 3px; + + border: 1px solid rgba(0, 0, 0, .1); + border-bottom: none; + box-shadow: inset 0 1px 0 #DADADA; +*/ + } + + .code-block { + padding: 10px; + margin: 0; + box-shadow: inset 0 0 6px rgba(0, 0, 0, .3); + } + + .linenums { + margin: 0; + margin-left: 10px; + } + + .frame-comments { + /* box-shadow: inset 0 0 6px rgba(0, 0, 0, .3); */ + /* border: 1px solid rgba(0, 0, 0, .2); */ + border-top: none; +/* + border-bottom-right-radius: 3px; + border-bottom-left-radius: 3px; +*/ + padding: 5px; + font-size: 12px; + background: #404040; + } + + .frame-comments.empty { + padding: 8px 15px; + } + + .frame-comments.empty:before { + content: "No comments for this stack frame."; + font-style: italic; + color: #828282; + } + + .frame-comment { + padding: 10px 5px; + color: #D2D2D2; + } + + .frame-comment:not(:last-child) { + border-bottom: 1px dotted rgba(0, 0, 0, .3); + } + + .frame-comment-context { + font-size: 10px; + font-weight: bold; + color: #86D2B6; + } + +.data-table-container label { + font-size: 16px; + font-weight: bold; + color: #ED591A; + margin: 10px 0; + padding: 10px 0; + + display: block; + margin-bottom: 5px; + padding-bottom: 5px; + border-bottom: 1px dotted rgba(0, 0, 0, .2); +} + .data-table { + width: 100%; + margin: 10px 0; + font: 13px consolas, monospace; + } + + .data-table thead { + display: none; + } + + .data-table tr { + padding: 5px 0; + } + + .data-table td:first-child { + width: 20%; + min-width: 130px; + overflow: hidden; + color: #463C54; + padding-right: 5px; + + } + + .data-table td:last-child { + width: 80%; + color:#999; + -ms-word-break: break-all; + word-break: break-all; + word-break: break-word; + -webkit-hyphens: auto; + -moz-hyphens: auto; + hyphens: auto; + } + + .data-table .empty { + color: rgba(0, 0, 0, .3); + font-style: italic; + } + +.handler { + padding: 10px; + font: 14px monospace; +} + +.handler.active { + color: #BBBBBB; + background: #989898; + font-weight: bold; +} + +/* prettify code style +Uses the Doxy theme as a base */ +pre .str, code .str { color: #E3B446; } /* string */ +pre .kwd, code .kwd { color: #DB613B; font-weight: bold; } /* keyword*/ +pre .com, code .com { color: #555; font-weight: bold; } /* comment */ +pre .typ, code .typ { color: #fff; } /* type */ +pre .lit, code .lit { color: #17CFB6; } /* literal */ +pre .pun, code .pun { color: #93a1a1; font-weight: bold; } /* punctuation */ +pre .pln, code .pln { color: #ccc; } /* plaintext */ +pre .tag, code .tag { color: #DB613B; } /* html/xml tag */ +pre .htm, code .htm { color: #dda0dd; } /* html tag */ +pre .xsl, code .xsl { color: #d0a0d0; } /* xslt tag */ +pre .atn, code .atn { color: #fff; font-weight: normal;} /* html/xml attribute name */ +pre .atv, code .atv { color: #E3B446; } /* html/xml attribute value */ +pre .dec, code .dec { color: #fff; } /* decimal */ +pre.prettyprint, code.prettyprint { + font-family: consolas, monospace; + background: #272727; + color: #929292; +} + pre.prettyprint { + white-space: pre-wrap; + } + + pre.prettyprint a, code.prettyprint a { + text-decoration:none; + } + + .linenums li.current{ + background: rgba(255, 255, 255, .05); + padding-top: 4px; + padding-left: 1px; + } + .linenums li.current.active { + background: rgba(255, 255, 255, .1); + } diff --git a/src/Illuminate/Exception/resources/pretty-template.php b/src/Illuminate/Exception/resources/pretty-template.php new file mode 100644 index 000000000000..979d64a7efc5 --- /dev/null +++ b/src/Illuminate/Exception/resources/pretty-template.php @@ -0,0 +1,195 @@ + + + + + + <?php echo $e($v->title) ?> + + + + +
+ +
+ +
+ + + frames as $i => $frame): ?> +
+
+ getClass() ?: '') ?> + getFunction() ?: '') ?> +
+ + + getFile(true) ?: '<#unknown>') ?>getLine() ?> + +
+ + +
+ +
+ +
+
+

+ name as $i => $nameSection): ?> + name) - 1): ?> + + + + + +

+

+ message) ?> +

+
+
+ + +
+ frames as $i => $frame): ?> + getLine(); ?> +
+
+ getFile() ?: '<#unknown>') ?> +
+ getFileLines($line - 8, 10); + $start = key($range) + 1; + $code = join("\n", $range); + ?> +
+ + + getComments(); + ?> +
+ $comment): ?> + +
+ + +
+ +
+ +
+ +
+ + +
+
+ tables as $label => $data): ?> +
+ + + + + + + + + + $value): ?> + + + + + +
KeyValue
+ + empty + +
+ +
+ + +
+ + handlers as $i => $handler): ?> +
+ . +
+ +
+ +
+
+ +
+
+ + + + + +