Valinor/tests/Unit/Mapper/Object/MethodObjectBuilderTest.php

190 lines
6.8 KiB
PHP
Raw Normal View History

2021-11-28 17:43:02 +01:00
<?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\ThrowableMessage;
2021-11-28 17:43:02 +01:00
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_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
2021-11-28 17:43:02 +01:00
{
return true;
2021-11-28 17:43:02 +01:00
}
};
$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`.');
2021-11-28 17:43:02 +01:00
$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(ThrowableMessage::class);
$this->expectExceptionCode(1337);
2021-11-28 17:43:02 +01:00
$this->expectExceptionMessage('some exception');
$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.');
2021-11-28 17:43:02 +01:00
$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));
2021-11-28 17:43:02 +01:00
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);
}
2021-11-28 17:43:02 +01:00
}
final class ObjectWithPrivateNativeConstructor
2021-11-28 17:43:02 +01:00
{
private function __construct()
2021-11-28 17:43:02 +01:00
{
}
}
final class ObjectWithConstructorThatThrowsException
{
public function __construct()
{
throw new RuntimeException('some exception', 1337);
2021-11-28 17:43:02 +01:00
}
}