Skip to content

Commit

Permalink
Fix proxies with return types in magic methods
Browse files Browse the repository at this point in the history
Co-authored-by: Jan Barášek <janbarasek@gmail.com>
Co-authored-by: Miloš Havlíček <miloshavlicek@gmail.com>
  • Loading branch information
3 people committed May 28, 2020
1 parent c72150a commit 0f00cb9
Show file tree
Hide file tree
Showing 9 changed files with 474 additions and 32 deletions.
109 changes: 77 additions & 32 deletions lib/Doctrine/Common/Proxy/ProxyGenerator.php
Original file line number Diff line number Diff line change
@@ -1,10 +1,10 @@
<?php
namespace Doctrine\Common\Proxy;

use Doctrine\Persistence\Mapping\ClassMetadata;
use Doctrine\Common\Proxy\Exception\InvalidArgumentException;
use Doctrine\Common\Proxy\Exception\UnexpectedValueException;
use Doctrine\Common\Util\ClassUtils;
use Doctrine\Persistence\Mapping\ClassMetadata;
use function array_map;
use function method_exists;

Expand Down Expand Up @@ -433,14 +433,24 @@ private function generateMagicGet(ClassMetadata $class)
$hasParentGet = false;
$returnReference = '';
$inheritDoc = '';
$name = '$name';
$parametersString = '$name';
$returnTypeHint = null;

if ($reflectionClass->hasMethod('__get')) {
$hasParentGet = true;
$inheritDoc = '{@inheritDoc}';
$hasParentGet = true;
$inheritDoc = '{@inheritDoc}';
$methodReflection = $reflectionClass->getMethod('__get');

if ($reflectionClass->getMethod('__get')->returnsReference()) {
if ($methodReflection->returnsReference()) {
$returnReference = '& ';
}

$methodParameters = $methodReflection->getParameters();
$name = '$' . $methodParameters[0]->getName();

$parametersString = $this->buildParametersString($methodReflection->getParameters());
$returnTypeHint = $this->getMethodReturnType($methodReflection);
}

if (empty($lazyPublicProperties) && ! $hasParentGet) {
Expand All @@ -450,42 +460,63 @@ private function generateMagicGet(ClassMetadata $class)
$magicGet = <<<EOT
/**
* $inheritDoc
* @param string \$name
* @param string $name
*/
public function {$returnReference}__get(\$name)
public function {$returnReference}__get($parametersString)$returnTypeHint
{
EOT;

if ( ! empty($lazyPublicProperties)) {
$magicGet .= sprintf(<<<'EOT'
if (\array_key_exists(%s, self::$lazyPropertiesNames)) {
$this->__initializer__ && $this->__initializer__->__invoke($this, '__get', [%s]);
EOT
, $name, $name);

if ($returnTypeHint === ': void') {
$magicGet .= "\n return;";
} else {
$magicGet .= "\n return \$this->$name;";
}

$magicGet .= <<<'EOT'
if (\array_key_exists($name, self::$lazyPropertiesNames)) {
$this->__initializer__ && $this->__initializer__->__invoke($this, '__get', [$name]);
return $this->$name;
}


EOT;
}

if ($hasParentGet) {
$magicGet .= <<<'EOT'
$this->__initializer__ && $this->__initializer__->__invoke($this, '__get', [$name]);
return parent::__get($name);

EOT;
$magicGet .= sprintf(<<<'EOT'
$this->__initializer__ && $this->__initializer__->__invoke($this, '__get', [%s]);
EOT
, $name);

if ($returnTypeHint === ': void') {
$magicGet .= sprintf(<<<'EOT'
parent::__get(%s);
return;
EOT
, $name);
} else {
$magicGet .= sprintf(<<<'EOT'
return parent::__get(%s);
EOT
, $name);
}
} else {
$magicGet .= <<<'EOT'
trigger_error(sprintf('Undefined property: %s::$%s', __CLASS__, $name), E_USER_NOTICE);
$magicGet .= sprintf(<<<EOT
trigger_error(sprintf('Undefined property: %%s::$%%s', __CLASS__, %s), E_USER_NOTICE);
EOT;
EOT
, $name);
}

$magicGet .= " }";

return $magicGet;
return $magicGet . "\n }";
}

/**
Expand All @@ -499,22 +530,31 @@ private function generateMagicSet(ClassMetadata $class)
{
$lazyPublicProperties = $this->getLazyLoadedPublicPropertiesNames($class);
$hasParentSet = $class->getReflectionClass()->hasMethod('__set');
$parametersString = '$name, $value';
$returnTypeHint = null;

if ($hasParentSet) {
$methodReflection = $class->getReflectionClass()->getMethod('__set');
$parametersString = $this->buildParametersString($methodReflection->getParameters());
$returnTypeHint = $this->getMethodReturnType($methodReflection);
}

if (empty($lazyPublicProperties) && ! $hasParentSet) {
return '';
}

$inheritDoc = $hasParentSet ? '{@inheritDoc}' : '';
$magicSet = <<<EOT
$magicSet = sprintf(<<<'EOT'
/**
* $inheritDoc
* @param string \$name
* @param mixed \$value
* %s
* @param string $name
* @param mixed $value
*/
public function __set(\$name, \$value)
public function __set(%s)%s
{

EOT;
EOT
, $inheritDoc, $parametersString, $returnTypeHint);

if ( ! empty($lazyPublicProperties)) {
$magicSet .= <<<'EOT'
Expand All @@ -540,9 +580,7 @@ public function __set(\$name, \$value)
$magicSet .= " \$this->\$name = \$value;";
}

$magicSet .= "\n }";

return $magicSet;
return $magicSet . "\n }";
}

/**
Expand All @@ -556,6 +594,14 @@ private function generateMagicIsset(ClassMetadata $class)
{
$lazyPublicProperties = $this->getLazyLoadedPublicPropertiesNames($class);
$hasParentIsset = $class->getReflectionClass()->hasMethod('__isset');
$parametersString = '$name';
$returnTypeHint = null;

if ($hasParentIsset) {
$methodReflection = $class->getReflectionClass()->getMethod('__isset');
$parametersString = $this->buildParametersString($methodReflection->getParameters());
$returnTypeHint = $this->getMethodReturnType($methodReflection);
}

if (empty($lazyPublicProperties) && ! $hasParentIsset) {
return '';
Expand All @@ -568,7 +614,7 @@ private function generateMagicIsset(ClassMetadata $class)
* @param string \$name
* @return boolean
*/
public function __isset(\$name)
public function __isset($parametersString)$returnTypeHint
{
EOT;
Expand All @@ -590,7 +636,6 @@ public function __isset(\$name)
$this->__initializer__ && $this->__initializer__->__invoke($this, '__isset', [$name]);
return parent::__isset($name);

EOT;
} else {
$magicIsset .= " return false;";
Expand Down
3 changes: 3 additions & 0 deletions phpstan.neon.dist
Original file line number Diff line number Diff line change
Expand Up @@ -37,6 +37,9 @@ parameters:
-
message: '#^Property Doctrine\\Tests\\Common\\Proxy\\ProxyLogicVoidReturnTypeTest::\$initializerCallbackMock \(callable&PHPUnit\\Framework\\MockObject\\MockObject\) does not accept PHPUnit\\Framework\\MockObject\\MockObject&stdClass\.$#'
path: '%currentWorkingDirectory%/tests/Doctrine/Tests/Common/Proxy/ProxyLogicVoidReturnTypeTest.php'
-
message: '#^Method Doctrine\\Tests\\Common\\Proxy\\MagicIssetClassWithInteger::__isset\(\) should return bool but returns int\.$#'
path: '%currentWorkingDirectory%/tests/Doctrine/Tests/Common/Proxy/MagicIssetClassWithInteger.php'

includes:
- vendor/phpstan/phpstan-phpunit/extension.neon
Expand Down
39 changes: 39 additions & 0 deletions tests/Doctrine/Tests/Common/Proxy/MagicGetClassWithScalarType.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
<?php

namespace Doctrine\Tests\Common\Proxy;

/**
* Test asset class
* @author Jan Barasek <jan@barasek.com>
*/
class MagicGetClassWithScalarType
{
/**
* @var string
*/
public $id = 'id';

/**
* @var string
*/
public $publicField = 'publicField';

/**
* @param string $name
*
* @return string
* @throws \BadMethodCallException
*/
public function __get(string $name): string
{
if ($name === 'test') {
return 'test';
}

if ($name === 'publicField' || $name === 'id') {
throw new \BadMethodCallException('Should never be called for "publicField" or "id"');
}

return 'not defined';
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
<?php

namespace Doctrine\Tests\Common\Proxy;

/**
* Test asset class
* @author Jan Barasek <jan@barasek.com>
*/
class MagicGetClassWithScalarTypeAndRenamedParameter
{
/**
* @var string
*/
public $id = 'id';

/**
* @var string
*/
public $publicField = 'publicField';

/**
* @param string $n
*
* @return string
* @throws \BadMethodCallException
*/
public function __get(string $n): string
{
if ($n === 'test') {
return 'test';
}

if ($n === 'publicField' || $n === 'id') {
throw new \BadMethodCallException('Should never be called for "publicField" or "id"');
}

return 'not defined';
}
}
21 changes: 21 additions & 0 deletions tests/Doctrine/Tests/Common/Proxy/MagicGetClassWithVoid.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
<?php

namespace Doctrine\Tests\Common\Proxy;

/**
* Magic class asset test with void return type for getter
*/
class MagicGetClassWithVoid
{

/**
* @param string $name
*
* @return void
* @throws \BadMethodCallException
*/
public function __get(string $name): void
{
return;
}
}
39 changes: 39 additions & 0 deletions tests/Doctrine/Tests/Common/Proxy/MagicIssetClassWithBoolean.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
<?php

namespace Doctrine\Tests\Common\Proxy;

/**
* Test asset class
* @author Jan Barasek <jan@barasek.com>
*/
class MagicIssetClassWithBoolean
{
/**
* @var string
*/
public $id = 'id';

/**
* @var string
*/
public $publicField = 'publicField';

/**
* @param string $name
*
* @return bool
* @throws \BadMethodCallException
*/
public function __isset(string $name): bool
{
if ('test' === $name) {
return true;
}

if ('publicField' === $name || 'id' === $name) {
throw new \BadMethodCallException('Should never be called for "publicField" or "id"');
}

return false;
}
}
39 changes: 39 additions & 0 deletions tests/Doctrine/Tests/Common/Proxy/MagicIssetClassWithInteger.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
<?php

namespace Doctrine\Tests\Common\Proxy;

/**
* Test asset class
* @author Jan Barasek <jan@barasek.com>
*/
class MagicIssetClassWithInteger
{
/**
* @var string
*/
public $id = 'id';

/**
* @var string
*/
public $publicField = 'publicField';

/**
* @param string $name
*
* @return int
* @throws \BadMethodCallException
*/
public function __isset(string $name): int
{
if ('test' === $name) {
return 1;
}

if ('publicField' === $name || 'id' === $name) {
throw new \BadMethodCallException('Should never be called for "publicField" or "id"');
}

return 0;
}
}
Loading

0 comments on commit 0f00cb9

Please sign in to comment.