diff --git a/phpstan-baseline.neon b/phpstan-baseline.neon index bcb0f598f..f20cbde25 100644 --- a/phpstan-baseline.neon +++ b/phpstan-baseline.neon @@ -16,7 +16,7 @@ parameters: path: src/Projection/Projection/ProjectionError.php - - message: "#^Parameter \\#2 \\$errorContext of class Patchlevel\\\\EventSourcing\\\\Projection\\\\Projection\\\\ProjectionError constructor expects array\\\\}\\>\\|null, mixed given\\.$#" + message: "#^Parameter \\#2 \\$errorContext of class Patchlevel\\\\EventSourcing\\\\Projection\\\\Projection\\\\ProjectionError constructor expects array\\\\}\\>\\|null, mixed given\\.$#" count: 1 path: src/Projection/Projection/Store/DoctrineStore.php @@ -30,6 +30,21 @@ parameters: count: 1 path: src/Projection/Projection/Store/ErrorContext.php + - + message: "#^Method Patchlevel\\\\EventSourcing\\\\Projection\\\\Projection\\\\Store\\\\ErrorContext\\:\\:transformTrace\\(\\) has parameter \\$trace with no value type specified in iterable type array\\.$#" + count: 1 + path: src/Projection/Projection/Store/ErrorContext.php + + - + message: "#^Method Patchlevel\\\\EventSourcing\\\\Projection\\\\Projection\\\\Store\\\\ErrorContext\\:\\:transformTrace\\(\\) return type has no value type specified in iterable type array\\.$#" + count: 1 + path: src/Projection/Projection/Store/ErrorContext.php + + - + message: "#^PHPDoc tag @var for variable \\$trace has no value type specified in iterable type array\\.$#" + count: 1 + path: src/Projection/Projection/Store/ErrorContext.php + - message: "#^Method Patchlevel\\\\EventSourcing\\\\Projection\\\\Projector\\\\InMemoryProjectorRepository\\:\\:projectors\\(\\) should return array\\ but returns array\\\\.$#" count: 1 diff --git a/src/Projection/Projection/Store/ErrorContext.php b/src/Projection/Projection/Store/ErrorContext.php index d4bb13962..0ed4cd046 100644 --- a/src/Projection/Projection/Store/ErrorContext.php +++ b/src/Projection/Projection/Store/ErrorContext.php @@ -6,9 +6,15 @@ use Throwable; +use function array_walk_recursive; +use function get_resource_type; +use function is_object; +use function is_resource; +use function sprintf; + /** * @psalm-type Trace = array{file?: string, line?: int, function?: string, class?: string, type?: string, args?: array} - * @psalm-type Context = array{message: string, code: int|string, file: string, line: int, trace: list} + * @psalm-type Context = array{class: class-string, message: string, code: int|string, file: string, line: int, trace: list} */ final class ErrorContext { @@ -29,11 +35,35 @@ public static function fromThrowable(Throwable $error): array private static function transformThrowable(Throwable $error): array { return [ + 'class' => $error::class, 'message' => $error->getMessage(), 'code' => $error->getCode(), 'file' => $error->getFile(), 'line' => $error->getLine(), - 'trace' => $error->getTrace(), + 'trace' => self::transformTrace($error->getTrace()), ]; } + + /** + * @param list $trace + * + * @return list + */ + private static function transformTrace(array $trace): array + { + array_walk_recursive($trace, static function (mixed &$value): void { + if (is_object($value)) { + $value = sprintf('object(%s)', $value::class); + } + + if (!is_resource($value)) { + return; + } + + $value = sprintf('resource(%s)', get_resource_type($value)); + }); + + /** @var list $trace */ + return $trace; + } } diff --git a/tests/Unit/Projection/Projection/ErrorContextTest.php b/tests/Unit/Projection/Projection/ErrorContextTest.php new file mode 100644 index 000000000..5dc596753 --- /dev/null +++ b/tests/Unit/Projection/Projection/ErrorContextTest.php @@ -0,0 +1,72 @@ +createException( + 'test', + new CustomId('test'), + $resource, + ['test' => [1, 2, 3]], + static fn () => null, + ), + ); + fclose($resource); + + $this->assertCount(1, $result); + $error = $result[0]; + + $this->assertSame(RuntimeException::class, $error['class']); + $this->assertSame('test', $error['message']); + $this->assertSame(0, $error['code']); + $this->assertSame(__FILE__, $error['file']); + $this->assertGreaterThan(0, count($error['trace'])); + $this->assertArrayHasKey(0, $error['trace']); + + $firstTrace = $error['trace'][0]; + + $this->assertArrayHasKey('file', $firstTrace); + $this->assertSame(__FILE__, $firstTrace['file'] ?? null); + $this->assertArrayHasKey('line', $firstTrace); + $this->assertSame('createException', $firstTrace['function'] ?? null); + $this->assertArrayHasKey('class', $firstTrace); + $this->assertSame(self::class, $firstTrace['class'] ?? null); + $this->assertArrayHasKey('type', $firstTrace); + $this->assertSame('->', $firstTrace['type'] ?? null); + $this->assertArrayHasKey('args', $firstTrace); + $this->assertSame([ + 'test', + 'object(Patchlevel\EventSourcing\Aggregate\CustomId)', + 'resource(stream)', + ['test' => [1, 2, 3]], + 'object(Closure)', + ], $firstTrace['args'] ?? null); + } + + /** @param resource $resource */ + private function createException( + string $message, + CustomId $id, + $resource, + array $array, + callable $callable, + ): RuntimeException { + return new RuntimeException($message); + } +}