Logger: Added handling for exceptions and log exceptions if they occur

This commit is contained in:
Igor Scheller 2020-04-19 20:41:38 +02:00 committed by msquare
parent 62f3e808bf
commit 64dc16e4d9
9 changed files with 148 additions and 24 deletions

View File

@ -7,6 +7,7 @@ use Engelsystem\Exceptions\Handlers\HandlerInterface;
use Engelsystem\Exceptions\Handlers\Legacy; use Engelsystem\Exceptions\Handlers\Legacy;
use Engelsystem\Exceptions\Handlers\LegacyDevelopment; use Engelsystem\Exceptions\Handlers\LegacyDevelopment;
use Engelsystem\Exceptions\Handlers\Whoops; use Engelsystem\Exceptions\Handlers\Whoops;
use Psr\Log\LoggerInterface;
use Whoops\Run as WhoopsRunner; use Whoops\Run as WhoopsRunner;
class ExceptionsServiceProvider extends ServiceProvider class ExceptionsServiceProvider extends ServiceProvider
@ -28,6 +29,7 @@ class ExceptionsServiceProvider extends ServiceProvider
$request = $this->app->get('request'); $request = $this->app->get('request');
$handler->setRequest($request); $handler->setRequest($request);
$this->addLogger($handler);
} }
/** /**
@ -55,4 +57,19 @@ class ExceptionsServiceProvider extends ServiceProvider
$this->app->instance('error.handler.development', $handler); $this->app->instance('error.handler.development', $handler);
$errorHandler->setHandler(Handler::ENV_DEVELOPMENT, $handler); $errorHandler->setHandler(Handler::ENV_DEVELOPMENT, $handler);
} }
/**
* @param Handler $handler
*/
protected function addLogger(Handler $handler)
{
foreach ($handler->getHandler() as $h) {
if (!method_exists($h, 'setLogger')) {
continue;
}
$log = $this->app->get(LoggerInterface::class);
$h->setLogger($log);
}
}
} }

View File

@ -3,10 +3,14 @@
namespace Engelsystem\Exceptions\Handlers; namespace Engelsystem\Exceptions\Handlers;
use Engelsystem\Http\Request; use Engelsystem\Http\Request;
use Psr\Log\LoggerInterface;
use Throwable; use Throwable;
class Legacy implements HandlerInterface class Legacy implements HandlerInterface
{ {
/** @var LoggerInterface */
protected $log;
/** /**
* @param Request $request * @param Request $request
* @param Throwable $e * @param Throwable $e
@ -29,6 +33,23 @@ class Legacy implements HandlerInterface
$e->getLine(), $e->getLine(),
json_encode($e->getTrace()) json_encode($e->getTrace())
)); ));
if (is_null($this->log)) {
return;
}
try {
$this->log->critical('', ['exception' => $e]);
} catch (Throwable $e) {
}
}
/**
* @param LoggerInterface $logger
*/
public function setLogger(LoggerInterface $logger)
{
$this->log = $logger;
} }
/** /**

View File

@ -6,6 +6,7 @@ use Engelsystem\Models\LogEntry;
use Psr\Log\AbstractLogger; use Psr\Log\AbstractLogger;
use Psr\Log\InvalidArgumentException; use Psr\Log\InvalidArgumentException;
use Psr\Log\LogLevel; use Psr\Log\LogLevel;
use Throwable;
class Logger extends AbstractLogger class Logger extends AbstractLogger
{ {
@ -34,15 +35,13 @@ class Logger extends AbstractLogger
/** /**
* Logs with an arbitrary level. * Logs with an arbitrary level.
* *
* @TODO: Implement $context['exception']
*
* @param mixed $level * @param mixed $level
* @param string $message * @param string $message
* @param array $context * @param array $context
* *
* @throws InvalidArgumentException * @throws InvalidArgumentException
*/ */
public function log($level, $message, array $context = []) public function log($level, $message, array $context = []): void
{ {
if (!$this->checkLevel($level)) { if (!$this->checkLevel($level)) {
throw new InvalidArgumentException('Unknown log level: ' . $level); throw new InvalidArgumentException('Unknown log level: ' . $level);
@ -50,6 +49,10 @@ class Logger extends AbstractLogger
$message = $this->interpolate($message, $context); $message = $this->interpolate($message, $context);
if (isset($context['exception']) && $context['exception'] instanceof Throwable) {
$message .= $this->formatException($context['exception']);
}
$this->log->create(['level' => $level, 'message' => $message]); $this->log->create(['level' => $level, 'message' => $message]);
} }
@ -60,7 +63,7 @@ class Logger extends AbstractLogger
* @param array $context * @param array $context
* @return string * @return string
*/ */
protected function interpolate($message, array $context = []) protected function interpolate($message, array $context = []): string
{ {
foreach ($context as $key => $val) { foreach ($context as $key => $val) {
// check that the value can be casted to string // check that the value can be casted to string
@ -75,11 +78,27 @@ class Logger extends AbstractLogger
return $message; return $message;
} }
/**
* @param Throwable $e
* @return string
*/
protected function formatException(Throwable $e): string
{
return sprintf(
implode(PHP_EOL, ['', 'Exception: %s', 'File: %s:%u', 'Code: %s', 'Trace:', '%s']),
$e->getMessage(),
$e->getFile(),
$e->getLine(),
$e->getCode(),
$e->getTraceAsString()
);
}
/** /**
* @param string $level * @param string $level
* @return bool * @return bool
*/ */
protected function checkLevel($level) protected function checkLevel($level): bool
{ {
return in_array($level, $this->allowedLevels); return in_array($level, $this->allowedLevels);
} }

View File

@ -19,7 +19,7 @@ class UserAwareLogger extends Logger
* *
* @throws InvalidArgumentException * @throws InvalidArgumentException
*/ */
public function log($level, $message, array $context = []) public function log($level, $message, array $context = []): void
{ {
if ($this->auth && ($user = $this->auth->user())) { if ($this->auth && ($user = $this->auth->user())) {
$message = sprintf('%s (%u): %s', $user->name, $user->id, $message); $message = sprintf('%s (%u): %s', $user->name, $user->id, $message);
@ -31,7 +31,7 @@ class UserAwareLogger extends Logger
/** /**
* @param Authenticator $auth * @param Authenticator $auth
*/ */
public function setAuth(Authenticator $auth) public function setAuth(Authenticator $auth): void
{ {
$this->auth = $auth; $this->auth = $auth;
} }

View File

@ -5,6 +5,7 @@ namespace Engelsystem\Test\Feature\Logger;
use Engelsystem\Logger\Logger; use Engelsystem\Logger\Logger;
use Engelsystem\Models\LogEntry; use Engelsystem\Models\LogEntry;
use Engelsystem\Test\Feature\ApplicationFeatureTest; use Engelsystem\Test\Feature\ApplicationFeatureTest;
use Exception;
use Psr\Log\InvalidArgumentException; use Psr\Log\InvalidArgumentException;
use Psr\Log\LoggerInterface; use Psr\Log\LoggerInterface;
use Psr\Log\LogLevel; use Psr\Log\LogLevel;
@ -144,6 +145,26 @@ class LoggerTest extends ApplicationFeatureTest
$logger->log('This log level should never be defined', 'Some message'); $logger->log('This log level should never be defined', 'Some message');
} }
/**
* @covers \Engelsystem\Logger\Logger::formatException
* @covers \Engelsystem\Logger\Logger::log
*/
public function testWithException()
{
$logger = $this->getLogger();
$logger->log(LogLevel::CRITICAL, 'Some random message', ['exception' => new Exception('Oops', 42)]);
$line = __LINE__ - 1;
$entry = $this->getLastEntry();
$this->assertStringContainsString('Some random message', $entry['message']);
$this->assertStringContainsString('Oops', $entry['message']);
$this->assertStringContainsString('42', $entry['message']);
$this->assertStringContainsString(__FILE__, $entry['message']);
$this->assertStringContainsString((string)$line, $entry['message']);
$this->assertStringContainsString(__FUNCTION__, $entry['message']);
}
/** /**
* @return array * @return array
*/ */

View File

@ -11,13 +11,14 @@ use Engelsystem\Exceptions\Handlers\Whoops;
use Engelsystem\Http\Request; use Engelsystem\Http\Request;
use Engelsystem\Test\Unit\ServiceProviderTest; use Engelsystem\Test\Unit\ServiceProviderTest;
use PHPUnit\Framework\MockObject\MockObject; use PHPUnit\Framework\MockObject\MockObject;
use Psr\Log\LoggerInterface;
class ExceptionsServiceProviderTest extends ServiceProviderTest class ExceptionsServiceProviderTest extends ServiceProviderTest
{ {
/** /**
* @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::addDevelopmentHandler() * @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::addDevelopmentHandler
* @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::addProductionHandler() * @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::addProductionHandler
* @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::register() * @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::register
*/ */
public function testRegister() public function testRegister()
{ {
@ -77,33 +78,57 @@ class ExceptionsServiceProviderTest extends ServiceProviderTest
} }
/** /**
* @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::boot() * @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::boot
* @covers \Engelsystem\Exceptions\ExceptionsServiceProvider::addLogger
*/ */
public function testBoot() public function testBoot()
{ {
/** @var HandlerInterface|MockObject $handlerImpl */
$handlerImpl = $this->getMockForAbstractClass(HandlerInterface::class);
/** @var Legacy|MockObject $loggingHandler */
$loggingHandler = $this->createMock(Legacy::class);
/** @var Handler|MockObject $handler */ /** @var Handler|MockObject $handler */
$handler = $this->createMock(Handler::class); $handler = $this->createMock(Handler::class);
/** @var Request|MockObject $request */ /** @var Request|MockObject $request */
$request = $this->createMock(Request::class); $request = $this->createMock(Request::class);
$handler->expects($this->once()) /** @var LoggerInterface|MockObject $log */
$log = $this->getMockForAbstractClass(LoggerInterface::class);
$handler->expects($this->exactly(2))
->method('setRequest') ->method('setRequest')
->with($request); ->with($request);
$handler->expects($this->exactly(2))
->method('getHandler')
->willReturnOnConsecutiveCalls([$handlerImpl], [$loggingHandler]);
$loggingHandler->expects($this->once())
->method('setLogger')
->with($log);
$app = $this->getApp(['get']); $app = $this->getApp(['get']);
$app->expects($this->exactly(2)) $app->expects($this->exactly(5))
->method('get') ->method('get')
->withConsecutive( ->withConsecutive(
['error.handler'], ['error.handler'],
['request'] ['request'],
['error.handler'],
['request'],
[LoggerInterface::class]
) )
->willReturnOnConsecutiveCalls( ->willReturnOnConsecutiveCalls(
$handler, $handler,
$request $request,
$handler,
$request,
$log
); );
$provider = new ExceptionsServiceProvider($app); $provider = new ExceptionsServiceProvider($app);
$provider->boot(); $provider->boot();
$provider->boot();
} }
} }

View File

@ -4,14 +4,16 @@ namespace Engelsystem\Test\Unit\Exceptions\Handlers;
use Engelsystem\Exceptions\Handlers\Legacy; use Engelsystem\Exceptions\Handlers\Legacy;
use Engelsystem\Http\Request; use Engelsystem\Http\Request;
use ErrorException;
use Exception; use Exception;
use PHPUnit\Framework\MockObject\MockObject; use PHPUnit\Framework\MockObject\MockObject;
use PHPUnit\Framework\TestCase; use PHPUnit\Framework\TestCase;
use Psr\Log\Test\TestLogger;
class LegacyTest extends TestCase class LegacyTest extends TestCase
{ {
/** /**
* @covers \Engelsystem\Exceptions\Handlers\Legacy::render() * @covers \Engelsystem\Exceptions\Handlers\Legacy::render
*/ */
public function testRender() public function testRender()
{ {
@ -27,22 +29,36 @@ class LegacyTest extends TestCase
} }
/** /**
* @covers \Engelsystem\Exceptions\Handlers\Legacy::report() * @covers \Engelsystem\Exceptions\Handlers\Legacy::report
* @covers \Engelsystem\Exceptions\Handlers\Legacy::stripBasePath() * @covers \Engelsystem\Exceptions\Handlers\Legacy::setLogger
* @covers \Engelsystem\Exceptions\Handlers\Legacy::stripBasePath
*/ */
public function testReport() public function testReport()
{ {
$handler = new Legacy(); $handler = new Legacy();
$exception = new Exception('Lorem Ipsum', 4242); $exception = new Exception('Lorem Ipsum', 4242);
$line = __LINE__ - 1; $line = __LINE__ - 1;
$exception2 = new Exception('Test Exception');
$exception3 = new Exception('Mor Exceptions!');
$logger = new TestLogger();
$logger2 = $this->createMock(TestLogger::class);
$logger2->expects($this->once())
->method('critical')
->willReturnCallback(function () {
throw new ErrorException();
});
$log = tempnam(sys_get_temp_dir(), 'engelsystem-log'); $logfile = tempnam(sys_get_temp_dir(), 'engelsystem-log');
$errorLog = ini_get('error_log'); $errorLog = ini_get('error_log');
ini_set('error_log', $log); ini_set('error_log', $logfile);
$handler->report($exception); $handler->report($exception);
$handler->setLogger($logger);
$handler->report($exception2);
$handler->setLogger($logger2);
$handler->report($exception3);
ini_set('error_log', $errorLog); ini_set('error_log', $errorLog);
$logContent = file_get_contents($log); $logContent = file_get_contents($logfile);
unset($log); unset($logfile);
$this->assertStringContainsString('4242', $logContent); $this->assertStringContainsString('4242', $logContent);
$this->assertStringContainsString('Lorem Ipsum', $logContent); $this->assertStringContainsString('Lorem Ipsum', $logContent);
@ -50,5 +66,10 @@ class LegacyTest extends TestCase
$this->assertStringContainsString((string)$line, $logContent); $this->assertStringContainsString((string)$line, $logContent);
$this->assertStringContainsString(__FUNCTION__, $logContent); $this->assertStringContainsString(__FUNCTION__, $logContent);
$this->assertStringContainsString(json_encode(__CLASS__), $logContent); $this->assertStringContainsString(json_encode(__CLASS__), $logContent);
$this->assertTrue($logger->hasRecordThatPasses(function (array $record) use ($exception2) {
$context = $record['context'];
return isset($context['exception']) && $context['exception'] === $exception2;
}, 'critical'));
} }
} }

View File

@ -4,8 +4,8 @@ namespace Engelsystem\Test\Unit\Logger;
use Engelsystem\Helpers\Authenticator; use Engelsystem\Helpers\Authenticator;
use Engelsystem\Logger\Logger; use Engelsystem\Logger\Logger;
use Engelsystem\Logger\UserAwareLogger;
use Engelsystem\Logger\LoggerServiceProvider; use Engelsystem\Logger\LoggerServiceProvider;
use Engelsystem\Logger\UserAwareLogger;
use Engelsystem\Test\Unit\ServiceProviderTest; use Engelsystem\Test\Unit\ServiceProviderTest;
use PHPUnit\Framework\MockObject\MockObject; use PHPUnit\Framework\MockObject\MockObject;
use Psr\Log\LoggerInterface; use Psr\Log\LoggerInterface;

View File

@ -19,7 +19,7 @@ class DevelopTest extends ExtensionTest
$extension = new Develop($config); $extension = new Develop($config);
$functions = $extension->getFunctions(); $functions = $extension->getFunctions();
$this->assertEquals($functions, []); $this->assertEquals([], $functions);
$config->set('environment', 'development'); $config->set('environment', 'development');
$functions = $extension->getFunctions(); $functions = $extension->getFunctions();