From 2c8a6cffc3220e99352ad958fe7cf06bf6f7690f Mon Sep 17 00:00:00 2001 From: Arnout Boks Date: Tue, 19 Mar 2024 15:01:30 +0100 Subject: [PATCH] Fix error when proxying a method with a mixed argument with a null default value Without this change, trying to proxy a Symfony\Component\Console\Command\Command would result in an InvalidArgumentException `Type "mixed" cannot be nullable` from the Laminas code generator. --- .../Generator/MethodGenerator.php | 2 +- .../Generator/MethodGeneratorTest.php | 22 +++++++++++++++++++ .../ClassWithNullDefaultMethodArguments.php | 13 +++++++++++ 3 files changed, 36 insertions(+), 1 deletion(-) create mode 100644 tests/ProxyManagerTestAsset/ClassWithNullDefaultMethodArguments.php diff --git a/src/ProxyManager/Generator/MethodGenerator.php b/src/ProxyManager/Generator/MethodGenerator.php index 1349c4e7..ae45ec64 100644 --- a/src/ProxyManager/Generator/MethodGenerator.php +++ b/src/ProxyManager/Generator/MethodGenerator.php @@ -69,7 +69,7 @@ public static function copyMethodSignature(MethodReflection $reflectionMethod): $parameter->setDefaultValue(new ValueGenerator($default, $reflectionParameter)); $type = $parameter->getType(); - if ($default->getValue() === null && strpos($type ?? '?', '?') !== 0 && strpos($type, '|') === false) { + if ($default->getValue() === null && strpos($type ?? '?', '?') !== 0 && strpos($type, '|') === false && $type !== 'mixed') { $parameter->setType('?' . $type); } } diff --git a/tests/ProxyManagerTest/Generator/MethodGeneratorTest.php b/tests/ProxyManagerTest/Generator/MethodGeneratorTest.php index d06ae613..9a8d53ee 100644 --- a/tests/ProxyManagerTest/Generator/MethodGeneratorTest.php +++ b/tests/ProxyManagerTest/Generator/MethodGeneratorTest.php @@ -10,6 +10,7 @@ use ProxyManager\Generator\MethodGenerator; use ProxyManagerTestAsset\BaseClass; use ProxyManagerTestAsset\ClassWithAbstractPublicMethod; +use ProxyManagerTestAsset\ClassWithNullDefaultMethodArguments; use ProxyManagerTestAsset\EmptyClass; use ProxyManagerTestAsset\ReturnTypeHintedClass; use ProxyManagerTestAsset\ScalarTypeHintedClass; @@ -134,6 +135,27 @@ public function scalarTypeHintedMethods(): array ]; } + /** + * @requires PHP 8.0 + */ + public function testGenerateMethodWithNullDefaultMixedArgument(): void + { + $method = MethodGenerator::fromReflectionWithoutBodyAndDocBlock(new MethodReflection( + ClassWithNullDefaultMethodArguments::class, + 'acceptMixed' + )); + + self::assertSame('acceptMixed', $method->getName()); + + $parameters = $method->getParameters(); + + self::assertCount(1, $parameters); + + $param = $parameters['param']; + + self::assertSame('mixed', $param->getType()); + } + public function testGenerateMethodWithVoidReturnTypeHinting(): void { $method = MethodGenerator::fromReflectionWithoutBodyAndDocBlock(new MethodReflection( diff --git a/tests/ProxyManagerTestAsset/ClassWithNullDefaultMethodArguments.php b/tests/ProxyManagerTestAsset/ClassWithNullDefaultMethodArguments.php new file mode 100644 index 00000000..be6f298f --- /dev/null +++ b/tests/ProxyManagerTestAsset/ClassWithNullDefaultMethodArguments.php @@ -0,0 +1,13 @@ +