Refactored ExceptionHandler

This commit is contained in:
Igor Scheller 2017-11-24 15:08:43 +01:00
parent 6eea072376
commit 25e434bce4
14 changed files with 716 additions and 141 deletions

View File

@ -23,7 +23,9 @@
"twbs/bootstrap": "^3.3" "twbs/bootstrap": "^3.3"
}, },
"require-dev": { "require-dev": {
"phpunit/phpunit": "^6.3" "filp/whoops": "^2.1",
"phpunit/phpunit": "^6.3",
"symfony/var-dumper": "^3.3"
}, },
"autoload": { "autoload": {
"psr-4": { "psr-4": {

View File

@ -2,7 +2,8 @@
use Engelsystem\Application; use Engelsystem\Application;
use Engelsystem\Config\Config; use Engelsystem\Config\Config;
use Engelsystem\Exceptions\BasicHandler as ExceptionHandler; use Engelsystem\Exceptions\Handler;
use Engelsystem\Exceptions\Handlers\HandlerInterface;
/** /**
* This file includes all needed functions, connects to the db etc. * This file includes all needed functions, connects to the db etc.
@ -32,7 +33,8 @@ date_default_timezone_set($app->get('config')->get('timezone'));
if (config('environment') == 'development') { if (config('environment') == 'development') {
$errorHandler = $app->get('error.handler'); $errorHandler = $app->get('error.handler');
$errorHandler->setEnvironment(ExceptionHandler::ENV_DEVELOPMENT); $errorHandler->setEnvironment(Handler::ENV_DEVELOPMENT);
$app->bind(HandlerInterface::class, 'error.handler.development');
ini_set('display_errors', true); ini_set('display_errors', true);
error_reporting(E_ALL); error_reporting(E_ALL);
} else { } else {

View File

@ -1,119 +0,0 @@
<?php
namespace Engelsystem\Exceptions;
use ErrorException;
use Throwable;
class BasicHandler extends Handler
{
/**
* Activate the error handler
*/
public function register()
{
set_error_handler([$this, 'errorHandler']);
set_exception_handler([$this, 'exceptionHandler']);
}
/**
* @param int $number
* @param string $message
* @param string $file
* @param int $line
*/
public function errorHandler($number, $message, $file, $line)
{
$exception = new ErrorException($message, 0, $number, $file, $line);
$this->exceptionHandler($exception);
}
/**
* @param Throwable $e
*/
public function exceptionHandler($e)
{
$this->handle(
$e->getCode(),
get_class($e) . ': ' . $e->getMessage(),
$e->getFile(),
$e->getLine(),
['exception' => $e]
);
}
/**
* @param int $number
* @param string $string
* @param string $file
* @param int $line
* @param array $context
* @param array $trace
*/
protected function handle($number, $string, $file, $line, $context = [], $trace = [])
{
error_log(sprintf('Exception: Number: %s, String: %s, File: %s:%u, Context: %s',
$number,
$string,
$file,
$line,
json_encode($context)
));
$file = $this->stripBasePath($file);
if ($this->environment == self::ENV_DEVELOPMENT) {
echo '<pre style="background-color:#333;color:#ccc;z-index:1000;position:fixed;bottom:1em;padding:1em;width:97%;max-height: 90%;overflow-y:auto;">';
echo sprintf('%s: (%s)' . PHP_EOL, ucfirst($type), $number);
var_export([
'string' => $string,
'file' => $file . ':' . $line,
'context' => $context,
'stacktrace' => $this->formatStackTrace($trace),
]);
echo '</pre>';
die();
}
echo 'An <del>un</del>expected error occurred, a team of untrained monkeys has been dispatched to deal with it.';
die();
}
/**
* @param array $stackTrace
* @return array
*/
protected function formatStackTrace($stackTrace)
{
$return = [];
foreach ($stackTrace as $trace) {
$path = '';
$line = '';
if (isset($trace['file']) && isset($trace['line'])) {
$path = $this->stripBasePath($trace['file']);
$line = $trace['line'];
}
$functionName = $trace['function'];
$return[] = [
'file' => $path . ':' . $line,
$functionName => $trace['args'],
];
}
return $return;
}
/**
* @param string $path
* @return string
*/
protected function stripBasePath($path)
{
$basePath = realpath(__DIR__ . '/../..') . '/';
return str_replace($basePath, '', $path);
}
}

View File

@ -3,14 +3,56 @@
namespace Engelsystem\Exceptions; namespace Engelsystem\Exceptions;
use Engelsystem\Container\ServiceProvider; use Engelsystem\Container\ServiceProvider;
use Engelsystem\Exceptions\Handlers\HandlerInterface;
use Engelsystem\Exceptions\Handlers\Legacy;
use Engelsystem\Exceptions\Handlers\LegacyDevelopment;
use Engelsystem\Exceptions\Handlers\Whoops;
use Whoops\Run as WhoopsRunner;
class ExceptionsServiceProvider extends ServiceProvider class ExceptionsServiceProvider extends ServiceProvider
{ {
public function register() public function register()
{ {
$errorHandler = $this->app->make(BasicHandler::class); $errorHandler = $this->app->make(Handler::class);
$errorHandler->register(); $this->addProductionHandler($errorHandler);
$this->addDevelopmentHandler($errorHandler);
$this->app->instance('error.handler', $errorHandler); $this->app->instance('error.handler', $errorHandler);
$this->app->bind(Handler::class, 'error.handler'); $this->app->bind(Handler::class, 'error.handler');
$errorHandler->register();
}
public function boot()
{
/** @var Handler $handler */
$handler = $this->app->get('error.handler');
$request = $this->app->get('request');
$handler->setRequest($request);
}
/**
* @param Handler $errorHandler
*/
protected function addProductionHandler($errorHandler)
{
$handler = $this->app->make(Legacy::class);
$this->app->instance('error.handler.production', $handler);
$errorHandler->setHandler(Handler::ENV_PRODUCTION, $handler);
$this->app->bind(HandlerInterface::class, 'error.handler.production');
}
/**
* @param Handler $errorHandler
*/
protected function addDevelopmentHandler($errorHandler)
{
$handler = $this->app->make(LegacyDevelopment::class);
if (class_exists(WhoopsRunner::class)) {
$handler = $this->app->make(Whoops::class);
}
$this->app->instance('error.handler.development', $handler);
$errorHandler->setHandler(Handler::ENV_DEVELOPMENT, $handler);
} }
} }

View File

@ -2,18 +2,29 @@
namespace Engelsystem\Exceptions; namespace Engelsystem\Exceptions;
abstract class Handler use Engelsystem\Exceptions\Handlers\HandlerInterface;
use Engelsystem\Http\Request;
use ErrorException;
use Throwable;
class Handler
{ {
/** @var string */ /** @var string */
protected $environment; protected $environment;
/** @var HandlerInterface[] */
protected $handler = [];
/** @var Request */
protected $request;
const ENV_PRODUCTION = 'prod'; const ENV_PRODUCTION = 'prod';
const ENV_DEVELOPMENT = 'dev'; const ENV_DEVELOPMENT = 'dev';
/** /**
* Handler constructor. * Handler constructor.
* *
* @param string $environment production|development * @param string $environment prod|dev
*/ */
public function __construct($environment = self::ENV_PRODUCTION) public function __construct($environment = self::ENV_PRODUCTION)
{ {
@ -25,6 +36,55 @@ abstract class Handler
*/ */
public function register() public function register()
{ {
set_error_handler([$this, 'errorHandler']);
set_exception_handler([$this, 'exceptionHandler']);
}
/**
* @param int $number
* @param string $message
* @param string $file
* @param int $line
*/
public function errorHandler($number, $message, $file, $line)
{
$exception = new ErrorException($message, 0, $number, $file, $line);
$this->exceptionHandler($exception);
}
/**
* @param Throwable $e
*/
public function exceptionHandler($e)
{
if (!$this->request instanceof Request) {
$this->request = new Request();
}
$handler = $this->handler[$this->environment];
$handler->report($e);
$handler->render($this->request, $e);
$this->die();
}
/**
* Exit the application
*
* @codeCoverageIgnore
* @param string $message
*/
protected function die($message = '')
{
echo $message;
die();
}
/**
* @return string
*/
public function getEnvironment()
{
return $this->environment;
} }
/** /**
@ -36,10 +96,40 @@ abstract class Handler
} }
/** /**
* @return string * @param string $environment
* @return HandlerInterface|HandlerInterface[]
*/ */
public function getEnvironment() public function getHandler($environment = null)
{ {
return $this->environment; if (!is_null($environment)) {
return $this->handler[$environment];
}
return $this->handler;
}
/**
* @param string $environment
* @param HandlerInterface $handler
*/
public function setHandler($environment, HandlerInterface $handler)
{
$this->handler[$environment] = $handler;
}
/**
* @return Request
*/
public function getRequest()
{
return $this->request;
}
/**
* @param Request $request
*/
public function setRequest(Request $request)
{
$this->request = $request;
} }
} }

View File

@ -0,0 +1,21 @@
<?php
namespace Engelsystem\Exceptions\Handlers;
use Engelsystem\Http\Request;
use Throwable;
interface HandlerInterface
{
/**
* @param Request $request
* @param Throwable $e
*/
public function render($request, Throwable $e);
/**
* @param Throwable $e
* @return
*/
public function report(Throwable $e);
}

View File

@ -0,0 +1,42 @@
<?php
namespace Engelsystem\Exceptions\Handlers;
use Engelsystem\Http\Request;
use Throwable;
class Legacy implements HandlerInterface
{
/**
* @param Request $request
* @param Throwable $e
*/
public function render($request, Throwable $e)
{
echo 'An <del>un</del>expected error occurred, a team of untrained monkeys has been dispatched to deal with it.';
}
/**
* @param Throwable $e
*/
public function report(Throwable $e)
{
error_log(sprintf('Exception: Code: %s, Message: %s, File: %s:%u, Trace: %s',
$e->getCode(),
$e->getMessage(),
$this->stripBasePath($e->getFile()),
$e->getLine(),
json_encode($e->getTrace())
));
}
/**
* @param string $path
* @return string
*/
protected function stripBasePath($path)
{
$basePath = realpath(__DIR__ . '/../../..') . '/';
return str_replace($basePath, '', $path);
}
}

View File

@ -0,0 +1,57 @@
<?php
namespace Engelsystem\Exceptions\Handlers;
use Engelsystem\Http\Request;
use Throwable;
class LegacyDevelopment extends Legacy
{
/**
* @param Request $request
* @param Throwable $e
*/
public function render($request, Throwable $e)
{
$file = $this->stripBasePath($e->getFile());
echo '<pre style="background-color:#333;color:#ccc;z-index:1000;position:fixed;bottom:1em;padding:1em;width:97%;max-height: 90%;overflow-y:auto;">';
echo sprintf('%s: (%s)' . PHP_EOL, get_class($e), $e->getCode());
$data = [
'string' => $e->getMessage(),
'file' => $file . ':' . $e->getLine(),
'stacktrace' => $this->formatStackTrace($e->getTrace()),
];
var_dump($data);
echo '</pre>';
}
/**
* @param array $stackTrace
* @return array
*/
protected function formatStackTrace($stackTrace)
{
$return = [];
$stackTrace = array_reverse($stackTrace);
foreach ($stackTrace as $trace) {
$path = '';
$line = '';
if (isset($trace['file']) && isset($trace['line'])) {
$path = $this->stripBasePath($trace['file']);
$line = $trace['line'];
}
$functionName = $trace['function'];
$return[] = [
'file' => $path . ':' . $line,
$functionName => isset($trace['args']) ? $trace['args'] : null,
];
}
return $return;
}
}

View File

@ -0,0 +1,85 @@
<?php
namespace Engelsystem\Exceptions\Handlers;
use Engelsystem\Application;
use Engelsystem\Container\Container;
use Engelsystem\Http\Request;
use Throwable;
use Whoops\Handler\JsonResponseHandler;
use Whoops\Handler\PrettyPageHandler;
use Whoops\Run as WhoopsRunner;
class Whoops extends Legacy implements HandlerInterface
{
/** @var Application */
protected $app;
public function __construct(Container $app)
{
$this->app = $app;
}
/**
* @param Request $request
* @param Throwable $e
*/
public function render($request, Throwable $e)
{
$whoops = $this->app->make(WhoopsRunner::class);
$handler = $this->getPrettyPageHandler($e);
$whoops->pushHandler($handler);
if ($request->isXmlHttpRequest()) {
$handler = $this->getJsonResponseHandler();
$whoops->pushHandler($handler);
}
echo $whoops->handleException($e);
}
/**
* @param Throwable $e
* @return PrettyPageHandler
*/
protected function getPrettyPageHandler(Throwable $e)
{
$handler = $this->app->make(PrettyPageHandler::class);
$handler->setPageTitle('Just another ' . get_class($e) . ' to fix :(');
$handler->setApplicationPaths([realpath(__DIR__ . '/../..')]);
$data = $this->getData();
$handler->addDataTable('Application', $data);
return $handler;
}
/**
* @return JsonResponseHandler
*/
protected function getJsonResponseHandler()
{
$handler = $this->app->make(JsonResponseHandler::class);
$handler->setJsonApi(true);
$handler->addTraceToOutput(true);
return $handler;
}
/**
* Aggregate application data
*
* @return array
*/
protected function getData()
{
global $user;
$data = [];
$data['user'] = $user;
$data['Booted'] = $this->app->isBooted();
return $data;
}
}

View File

@ -2,30 +2,108 @@
namespace Engelsystem\Test\Unit\Exceptions; namespace Engelsystem\Test\Unit\Exceptions;
use Engelsystem\Exceptions\BasicHandler as ExceptionHandler;
use Engelsystem\Exceptions\ExceptionsServiceProvider; use Engelsystem\Exceptions\ExceptionsServiceProvider;
use Engelsystem\Exceptions\Handler; use Engelsystem\Exceptions\Handler;
use Engelsystem\Exceptions\Handlers\HandlerInterface;
use Engelsystem\Exceptions\Handlers\Legacy;
use Engelsystem\Exceptions\Handlers\LegacyDevelopment;
use Engelsystem\Exceptions\Handlers\Whoops;
use Engelsystem\Http\Request;
use Engelsystem\Test\Unit\ServiceProviderTest; use Engelsystem\Test\Unit\ServiceProviderTest;
use PHPUnit_Framework_MockObject_MockObject; use PHPUnit_Framework_MockObject_MockObject as MockObject;
class ExceptionsServiceProviderTest extends ServiceProviderTest class ExceptionsServiceProviderTest extends ServiceProviderTest
{ {
/** /**
* @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::register() * @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::register()
* @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::addProductionHandler()
* @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::addDevelopmentHandler()
*/ */
public function testRegister() public function testRegister()
{ {
/** @var PHPUnit_Framework_MockObject_MockObject|ExceptionHandler $exceptionHandler */
$exceptionHandler = $this->getMockBuilder(ExceptionHandler::class)
->getMock();
$app = $this->getApp(['make', 'instance', 'bind']); $app = $this->getApp(['make', 'instance', 'bind']);
$this->setExpects($app, 'make', [ExceptionHandler::class], $exceptionHandler); /** @var MockObject|Handler $handler */
$this->setExpects($app, 'instance', ['error.handler', $exceptionHandler]); $handler = $this->createMock(Handler::class);
$this->setExpects($app, 'bind', [Handler::class, 'error.handler']); $this->setExpects($handler, 'register');
/** @var Legacy|MockObject $legacyHandler */
$legacyHandler = $this->createMock(Legacy::class);
/** @var LegacyDevelopment|MockObject $developmentHandler */
$developmentHandler = $this->createMock(LegacyDevelopment::class);
$whoopsHandler = $this->getMockBuilder(Whoops::class)
->setConstructorArgs([$app])
->getMock();
$app->expects($this->exactly(3))
->method('instance')
->withConsecutive(
['error.handler.production', $legacyHandler],
['error.handler.development', $whoopsHandler],
['error.handler', $handler]
);
$app->expects($this->exactly(4))
->method('make')
->withConsecutive(
[Handler::class],
[Legacy::class],
[LegacyDevelopment::class],
[Whoops::class]
)
->willReturnOnConsecutiveCalls(
$handler,
$legacyHandler,
$developmentHandler,
$whoopsHandler
);
$app->expects($this->exactly(2))
->method('bind')
->withConsecutive(
[HandlerInterface::class, 'error.handler.production'],
[Handler::class, 'error.handler']
);
$handler->expects($this->exactly(2))
->method('setHandler')
->withConsecutive(
[Handler::ENV_PRODUCTION, $legacyHandler],
[Handler::ENV_DEVELOPMENT, $whoopsHandler]
);
$serviceProvider = new ExceptionsServiceProvider($app); $serviceProvider = new ExceptionsServiceProvider($app);
$serviceProvider->register(); $serviceProvider->register();
} }
/**
* @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::boot()
*/
public function testBoot()
{
/** @var MockObject|Handler $handler */
$handler = $this->createMock(Handler::class);
/** @var MockObject|Request $request */
$request = $this->createMock(Request::class);
$handler->expects($this->once())
->method('setRequest')
->with($request);
$app = $this->getApp(['get']);
$app->expects($this->exactly(2))
->method('get')
->withConsecutive(
['error.handler'],
['request']
)
->willReturnOnConsecutiveCalls(
$handler,
$request
);
$provider = new ExceptionsServiceProvider($app);
$provider->boot();
}
} }

View File

@ -3,6 +3,10 @@
namespace Engelsystem\Test\Unit\Exceptions; namespace Engelsystem\Test\Unit\Exceptions;
use Engelsystem\Exceptions\Handler; use Engelsystem\Exceptions\Handler;
use Engelsystem\Exceptions\Handlers\HandlerInterface;
use Engelsystem\Http\Request;
use ErrorException;
use Exception;
use PHPUnit\Framework\TestCase; use PHPUnit\Framework\TestCase;
use PHPUnit_Framework_MockObject_MockObject as Mock; use PHPUnit_Framework_MockObject_MockObject as Mock;
@ -10,14 +14,80 @@ class HandlerTest extends TestCase
{ {
/** /**
* @covers \Engelsystem\Exceptions\Handler::__construct() * @covers \Engelsystem\Exceptions\Handler::__construct()
*/
public function testCreate()
{
/** @var Handler|Mock $handler */
$handler = new Handler();
$this->assertInstanceOf(Handler::class, $handler);
$this->assertEquals(Handler::ENV_PRODUCTION, $handler->getEnvironment());
$anotherHandler = new Handler(Handler::ENV_DEVELOPMENT);
$this->assertEquals(Handler::ENV_DEVELOPMENT, $anotherHandler->getEnvironment());
}
/**
* @covers \Engelsystem\Exceptions\Handler::errorHandler()
*/
public function testErrorHandler()
{
/** @var Handler|Mock $handler */
$handler = $this->getMockBuilder(Handler::class)
->setMethods(['exceptionHandler'])
->getMock();
$handler->expects($this->once())
->method('exceptionHandler')
->with($this->isInstanceOf(ErrorException::class));
$handler->errorHandler(1, 'Foo and bar!', '/lo/rem.php', 123);
}
/**
* @covers \Engelsystem\Exceptions\Handler::exceptionHandler()
*/
public function testExceptionHandler()
{
$exception = new Exception();
/** @var HandlerInterface|Mock $handlerMock */
$handlerMock = $this->getMockForAbstractClass(HandlerInterface::class);
$handlerMock->expects($this->once())
->method('report')
->with($exception);
$handlerMock->expects($this->once())
->method('render')
->with($this->isInstanceOf(Request::class), $exception);
/** @var Handler|Mock $handler */
$handler = $this->getMockBuilder(Handler::class)
->setMethods(['die'])
->getMock();
$handler->expects($this->once())
->method('die');
$handler->setHandler(Handler::ENV_PRODUCTION, $handlerMock);
$handler->exceptionHandler($exception);
}
/**
* @covers \Engelsystem\Exceptions\Handler::register() * @covers \Engelsystem\Exceptions\Handler::register()
*/ */
public function testRegister() public function testRegister()
{ {
/** @var Handler|Mock $handler */ /** @var Handler|Mock $handler */
$handler = $this->getMockForAbstractClass(Handler::class); $handler = $this->getMockForAbstractClass(Handler::class);
$this->assertInstanceOf(Handler::class, $handler);
$handler->register(); $handler->register();
set_error_handler($errorHandler = set_error_handler('var_dump'));
$this->assertEquals($handler, array_shift($errorHandler));
set_exception_handler($exceptionHandler = set_error_handler('var_dump'));
$this->assertEquals($handler, array_shift($exceptionHandler));
restore_error_handler();
restore_exception_handler();
} }
/** /**
@ -26,8 +96,7 @@ class HandlerTest extends TestCase
*/ */
public function testEnvironment() public function testEnvironment()
{ {
/** @var Handler|Mock $handler */ $handler = new Handler();
$handler = $this->getMockForAbstractClass(Handler::class);
$handler->setEnvironment(Handler::ENV_DEVELOPMENT); $handler->setEnvironment(Handler::ENV_DEVELOPMENT);
$this->assertEquals(Handler::ENV_DEVELOPMENT, $handler->getEnvironment()); $this->assertEquals(Handler::ENV_DEVELOPMENT, $handler->getEnvironment());
@ -35,4 +104,37 @@ class HandlerTest extends TestCase
$handler->setEnvironment(Handler::ENV_PRODUCTION); $handler->setEnvironment(Handler::ENV_PRODUCTION);
$this->assertEquals(Handler::ENV_PRODUCTION, $handler->getEnvironment()); $this->assertEquals(Handler::ENV_PRODUCTION, $handler->getEnvironment());
} }
/**
* @covers \Engelsystem\Exceptions\Handler::setHandler()
* @covers \Engelsystem\Exceptions\Handler::getHandler()
*/
public function testHandler()
{
$handler = new Handler();
/** @var HandlerInterface|Mock $devHandler */
$devHandler = $this->getMockForAbstractClass(HandlerInterface::class);
/** @var HandlerInterface|Mock $prodHandler */
$prodHandler = $this->getMockForAbstractClass(HandlerInterface::class);
$handler->setHandler(Handler::ENV_DEVELOPMENT, $devHandler);
$handler->setHandler(Handler::ENV_PRODUCTION, $prodHandler);
$this->assertEquals($devHandler, $handler->getHandler(Handler::ENV_DEVELOPMENT));
$this->assertEquals($prodHandler, $handler->getHandler(Handler::ENV_PRODUCTION));
$this->assertCount(2, $handler->getHandler());
}
/**
* @covers \Engelsystem\Exceptions\Handler::setRequest()
* @covers \Engelsystem\Exceptions\Handler::getRequest()
*/
public function testRequest()
{
$handler = new Handler();
/** @var Request|Mock $request */
$request = $this->createMock(Request::class);
$handler->setRequest($request);
$this->assertEquals($request, $handler->getRequest());
}
} }

View File

@ -0,0 +1,35 @@
<?php
namespace Engelsystem\Test\Unit\Exceptions\handlers;
use Engelsystem\Exceptions\Handlers\LegacyDevelopment;
use Engelsystem\Http\Request;
use ErrorException;
use PHPUnit\Framework\TestCase;
use PHPUnit_Framework_MockObject_MockObject as Mock;
class LegacyDevelopmentTest extends TestCase
{
/**
* @covers \Engelsystem\Exceptions\Handlers\LegacyDevelopment::render()
* @covers \Engelsystem\Exceptions\Handlers\LegacyDevelopment::formatStackTrace()
*/
public function testRender()
{
$handler = new LegacyDevelopment();
/** @var Request|Mock $request */
$request = $this->createMock(Request::class);
$exception = new ErrorException('Lorem Ipsum', 4242, 1, 'foo.php', 9999);
$regex = sprintf(
'%%<pre.*>.*ErrorException.*4242.*Lorem Ipsum.*%s.*%s.*%s.*</pre>%%is',
'foo.php',
9999,
__FUNCTION__
);
$this->expectOutputRegex($regex);
$handler->render($request, $exception);
}
}

View File

@ -0,0 +1,55 @@
<?php
namespace Engelsystem\Test\Unit\Exceptions\handlers;
use Engelsystem\Exceptions\Handlers\Legacy;
use Engelsystem\Http\Request;
use Exception;
use PHPUnit\Framework\TestCase;
use PHPUnit_Framework_MockObject_MockObject as Mock;
class LegacyTest extends TestCase
{
/**
* @covers \Engelsystem\Exceptions\Handlers\Legacy::render()
*/
public function testRender()
{
$handler = new Legacy();
/** @var Request|Mock $request */
$request = $this->createMock(Request::class);
/** @var Exception|Mock $exception */
$exception = $this->createMock(Exception::class);
$this->expectOutputRegex('/.*error occurred.*/i');
$handler->render($request, $exception);
}
/**
* @covers \Engelsystem\Exceptions\Handlers\Legacy::report()
* @covers \Engelsystem\Exceptions\Handlers\Legacy::stripBasePath()
*/
public function testReport()
{
$handler = new Legacy();
$exception = new Exception('Lorem Ipsum', 4242);
$line = __LINE__ - 1;
$log = tempnam(sys_get_temp_dir(), 'engelsystem-log');
$errorLog = ini_get('error_log');
ini_set('error_log', $log);
$handler->report($exception);
ini_set('error_log', $errorLog);
$logContent = file_get_contents($log);
unset($log);
$this->assertContains('4242', $logContent);
$this->assertContains('Lorem Ipsum', $logContent);
$this->assertContains(basename(__FILE__), $logContent);
$this->assertContains((string)$line, $logContent);
$this->assertContains(__FUNCTION__, $logContent);
$this->assertContains(json_encode(__CLASS__), $logContent);
}
}

View File

@ -0,0 +1,83 @@
<?php
namespace Engelsystem\Test\Unit\Exceptions\handlers;
use Engelsystem\Application;
use Engelsystem\Exceptions\Handlers\Whoops;
use Engelsystem\Http\Request;
use Exception;
use PHPUnit\Framework\TestCase;
use PHPUnit_Framework_MockObject_MockObject as Mock;
use Whoops\Handler\JsonResponseHandler;
use Whoops\Handler\PrettyPageHandler;
use Whoops\Run as WhoopsRunner;
use Whoops\RunInterface as WhoopsRunnerInterface;
class WhoopsTest extends TestCase
{
/**
* @covers \Engelsystem\Exceptions\Handlers\Whoops
*/
public function testRender()
{
/** @var Application|Mock $app */
$app = $this->createMock(Application::class);
/** @var Request|Mock $request */
$request = $this->createMock(Request::class);
$request->expects($this->once())
->method('isXmlHttpRequest')
->willReturn(true);
/** @var WhoopsRunnerInterface|Mock $whoopsRunner */
$whoopsRunner = $this->getMockForAbstractClass(WhoopsRunnerInterface::class);
/** @var PrettyPageHandler|Mock $prettyPageHandler */
$prettyPageHandler = $this->createMock(PrettyPageHandler::class);
$prettyPageHandler
->expects($this->atLeastOnce())
->method('setApplicationPaths');
$prettyPageHandler
->expects($this->once())
->method('setApplicationPaths');
$prettyPageHandler
->expects($this->once())
->method('addDataTable');
/** @var JsonResponseHandler|Mock $jsonResponseHandler */
$jsonResponseHandler = $this->createMock(JsonResponseHandler::class);
$jsonResponseHandler->expects($this->once())
->method('setJsonApi')
->with(true);
$jsonResponseHandler->expects($this->once())
->method('addTraceToOutput')
->with(true);
/** @var Exception|Mock $exception */
$exception = $this->createMock(Exception::class);
$app->expects($this->exactly(3))
->method('make')
->withConsecutive(
[WhoopsRunner::class],
[PrettyPageHandler::class],
[JsonResponseHandler::class]
)
->willReturnOnConsecutiveCalls(
$whoopsRunner,
$prettyPageHandler,
$jsonResponseHandler
);
$whoopsRunner
->expects($this->exactly(2))
->method('pushHandler')
->withConsecutive(
[$prettyPageHandler],
[$jsonResponseHandler]
);
$whoopsRunner
->expects($this->once())
->method('handleException')
->with($exception);
$handler = new Whoops($app);
$handler->render($request, $exception);
}
}