mirror of
https://github.com/danog/Valinor.git
synced 2025-01-10 06:38:24 +01:00
6ce1a439ad
/!\ This change fixes a security issue. Userland exception thrown in a constructor will not be automatically caught by the mapper anymore. This prevents messages with sensible information from reaching the final user — for instance an SQL exception showing a part of a query. To allow exceptions to be considered as safe, the new method `MapperBuilder::filterExceptions()` must be used, with caution. ```php final class SomeClass { public function __construct(private string $value) { \Webmozart\Assert\Assert::startsWith($value, 'foo_'); } } try { (new \CuyZ\Valinor\MapperBuilder()) ->filterExceptions(function (Throwable $exception) { if ($exception instanceof \Webmozart\Assert\InvalidArgumentException) { return \CuyZ\Valinor\Mapper\Tree\Message\ThrowableMessage::from($exception); } // If the exception should not be caught by this library, it // must be thrown again. throw $exception; }) ->mapper() ->map(SomeClass::class, 'bar_baz'); } catch (\CuyZ\Valinor\Mapper\MappingError $exception) { // Should print something similar to: // > Expected a value to start with "foo_". Got: "bar_baz" echo $exception->node()->messages()[0]; } ```
202 lines
7.2 KiB
PHP
202 lines
7.2 KiB
PHP
<?php
|
|
|
|
declare(strict_types=1);
|
|
|
|
namespace CuyZ\Valinor\Tests\Unit\Mapper\Object;
|
|
|
|
use CuyZ\Valinor\Mapper\Object\Exception\ConstructorMethodIsNotPublic;
|
|
use CuyZ\Valinor\Mapper\Object\Exception\ConstructorMethodIsNotStatic;
|
|
use CuyZ\Valinor\Mapper\Object\Exception\InvalidConstructorMethodClassReturnType;
|
|
use CuyZ\Valinor\Mapper\Object\Exception\MethodNotFound;
|
|
use CuyZ\Valinor\Mapper\Object\MethodObjectBuilder;
|
|
use CuyZ\Valinor\Mapper\Tree\Message\UserlandError;
|
|
use CuyZ\Valinor\Tests\Fake\Definition\FakeClassDefinition;
|
|
use PHPUnit\Framework\TestCase;
|
|
use ReflectionClass;
|
|
use RuntimeException;
|
|
use stdClass;
|
|
|
|
use function get_class;
|
|
|
|
final class MethodObjectBuilderTest extends TestCase
|
|
{
|
|
public function test_build_object_with_constructor_returns_correct_object(): void
|
|
{
|
|
$object = new class ('foo', 'bar') {
|
|
public string $valueA;
|
|
|
|
public string $valueB;
|
|
|
|
public string $valueC;
|
|
|
|
public function __construct(
|
|
string $valueA,
|
|
string $valueB,
|
|
string $valueC = 'Some parameter default value'
|
|
) {
|
|
$this->valueA = $valueA;
|
|
$this->valueB = $valueB;
|
|
$this->valueC = $valueC;
|
|
}
|
|
};
|
|
|
|
$class = FakeClassDefinition::fromReflection(new ReflectionClass($object));
|
|
$objectBuilder = new MethodObjectBuilder($class, '__construct');
|
|
$result = $objectBuilder->build([
|
|
'valueA' => 'valueA',
|
|
'valueB' => 'valueB',
|
|
'valueC' => 'valueC',
|
|
]);
|
|
|
|
self::assertSame('valueA', $result->valueA); // @phpstan-ignore-line
|
|
self::assertSame('valueB', $result->valueB); // @phpstan-ignore-line
|
|
self::assertSame('valueC', $result->valueC); // @phpstan-ignore-line
|
|
}
|
|
|
|
public function test_signature_is_method_signature(): void
|
|
{
|
|
$object = new class () {
|
|
public function __construct()
|
|
{
|
|
}
|
|
};
|
|
|
|
$class = FakeClassDefinition::fromReflection(new ReflectionClass($object));
|
|
$objectBuilder = new MethodObjectBuilder($class, '__construct');
|
|
|
|
self::assertSame('Signature::__construct', $objectBuilder->signature());
|
|
}
|
|
|
|
public function test_not_existing_method_throws_exception(): void
|
|
{
|
|
$this->expectException(MethodNotFound::class);
|
|
$this->expectExceptionCode(1634044209);
|
|
$this->expectExceptionMessage('Method `notExistingMethod` was not found in class `stdClass`.');
|
|
|
|
$class = FakeClassDefinition::fromReflection(new ReflectionClass(stdClass::class));
|
|
new MethodObjectBuilder($class, 'notExistingMethod');
|
|
}
|
|
|
|
public function test_invalid_constructor_method_throws_exception(): void
|
|
{
|
|
$this->expectException(ConstructorMethodIsNotStatic::class);
|
|
$this->expectExceptionCode(1634044370);
|
|
$this->expectExceptionMessage('Invalid constructor method `Signature::invalidConstructor`: it is neither the constructor nor a static constructor.');
|
|
|
|
$object = new class () {
|
|
public function invalidConstructor(): void
|
|
{
|
|
}
|
|
};
|
|
|
|
$class = FakeClassDefinition::fromReflection(new ReflectionClass($object));
|
|
new MethodObjectBuilder($class, 'invalidConstructor');
|
|
}
|
|
|
|
public function test_invalid_constructor_method_return_type_throws_exception(): void
|
|
{
|
|
$object = new class () {
|
|
public static function invalidConstructor(): bool
|
|
{
|
|
return true;
|
|
}
|
|
};
|
|
|
|
$this->expectException(InvalidConstructorMethodClassReturnType::class);
|
|
$this->expectExceptionCode(1638094499);
|
|
$this->expectExceptionMessage('Method `Signature::invalidConstructor` must return `' . get_class($object) . '` to be a valid constructor but returns `bool`.');
|
|
|
|
$class = FakeClassDefinition::fromReflection(new ReflectionClass($object));
|
|
new MethodObjectBuilder($class, 'invalidConstructor');
|
|
}
|
|
|
|
public function test_invalid_constructor_method_class_return_type_throws_exception(): void
|
|
{
|
|
$object = new class () {
|
|
public static function invalidConstructor(): stdClass
|
|
{
|
|
return new stdClass();
|
|
}
|
|
};
|
|
|
|
$this->expectException(InvalidConstructorMethodClassReturnType::class);
|
|
$this->expectExceptionCode(1638094499);
|
|
$this->expectExceptionMessage('Method `Signature::invalidConstructor` must return `' . get_class($object) . '` to be a valid constructor but returns `stdClass`.');
|
|
|
|
$class = FakeClassDefinition::fromReflection(new ReflectionClass($object));
|
|
new MethodObjectBuilder($class, 'invalidConstructor');
|
|
}
|
|
|
|
public function test_exception_thrown_by_constructor_is_caught_and_wrapped(): void
|
|
{
|
|
$class = FakeClassDefinition::fromReflection(new ReflectionClass(ObjectWithConstructorThatThrowsException::class));
|
|
$objectBuilder = new MethodObjectBuilder($class, '__construct');
|
|
|
|
$this->expectException(UserlandError::class);
|
|
|
|
$objectBuilder->build([]);
|
|
}
|
|
|
|
public function test_constructor_builder_for_class_with_private_constructor_throws_exception(): void
|
|
{
|
|
$this->expectException(ConstructorMethodIsNotPublic::class);
|
|
$this->expectExceptionCode(1630937169);
|
|
$this->expectExceptionMessage('The constructor of the class `' . ObjectWithPrivateNativeConstructor::class . '` is not public.');
|
|
|
|
$class = FakeClassDefinition::fromReflection(new ReflectionClass(ObjectWithPrivateNativeConstructor::class));
|
|
new MethodObjectBuilder($class, '__construct');
|
|
}
|
|
|
|
public function test_constructor_builder_for_class_with_private_named_constructor_throws_exception(): void
|
|
{
|
|
$classWithPrivateNativeConstructor = new class () {
|
|
// @phpstan-ignore-next-line
|
|
private static function someConstructor(): void
|
|
{
|
|
}
|
|
};
|
|
|
|
$this->expectException(ConstructorMethodIsNotPublic::class);
|
|
$this->expectExceptionCode(1630937169);
|
|
$this->expectExceptionMessage('The named constructor `Signature::someConstructor` is not public.');
|
|
|
|
$class = FakeClassDefinition::fromReflection(new ReflectionClass($classWithPrivateNativeConstructor));
|
|
new MethodObjectBuilder($class, 'someConstructor');
|
|
}
|
|
|
|
public function test_arguments_instance_stays_the_same(): void
|
|
{
|
|
$class = new class ('foo') {
|
|
public string $string;
|
|
|
|
public function __construct(string $string)
|
|
{
|
|
$this->string = $string;
|
|
}
|
|
};
|
|
$class = FakeClassDefinition::fromReflection(new ReflectionClass($class));
|
|
|
|
$objectBuilder = new MethodObjectBuilder($class, '__construct');
|
|
|
|
$argumentsA = $objectBuilder->describeArguments();
|
|
$argumentsB = $objectBuilder->describeArguments();
|
|
|
|
self::assertSame($argumentsA, $argumentsB);
|
|
}
|
|
}
|
|
|
|
final class ObjectWithPrivateNativeConstructor
|
|
{
|
|
private function __construct()
|
|
{
|
|
}
|
|
}
|
|
|
|
final class ObjectWithConstructorThatThrowsException
|
|
{
|
|
public function __construct()
|
|
{
|
|
throw new RuntimeException('some exception', 1337);
|
|
}
|
|
}
|