1
0
mirror of synced 2025-01-25 01:31:40 +03:00

Merge pull request #1207 from Ocramius/hotfix/embedded-classes-reflection-new-instance-creation

Embedded classes reflection new instance creation with internal PHP classes
This commit is contained in:
Steve Müller 2014-12-05 14:24:23 +01:00
commit 38650b748d
3 changed files with 190 additions and 11 deletions

View File

@ -19,6 +19,9 @@
namespace Doctrine\ORM\Mapping;
use Doctrine\Instantiator\Instantiator;
use ReflectionProperty;
/**
* Acts as a proxy to a nested Property structure, making it look like
* just a single scalar property.
@ -28,36 +31,62 @@ namespace Doctrine\ORM\Mapping;
*
* TODO: Move this class into Common\Reflection
*/
class ReflectionEmbeddedProperty
class ReflectionEmbeddedProperty extends ReflectionProperty
{
/**
* @var ReflectionProperty
*/
private $parentProperty;
private $childProperty;
private $class;
public function __construct($parentProperty, $childProperty, $class)
/**
* @var ReflectionProperty
*/
private $childProperty;
/**
* @var Instantiator|null
*/
private $instantiator;
/**
* @param ReflectionProperty $parentProperty
* @param ReflectionProperty $childProperty
* @param string $class
*/
public function __construct(ReflectionProperty $parentProperty, ReflectionProperty $childProperty, $class)
{
$this->parentProperty = $parentProperty;
$this->childProperty = $childProperty;
$this->class = $class;
parent::__construct($childProperty->getDeclaringClass()->getName(), $childProperty->getName());
}
public function getValue($object)
/**
* {@inheritDoc}
*/
public function getValue($object = null)
{
$embeddedObject = $this->parentProperty->getValue($object);
if ($embeddedObject === null) {
if (null === $embeddedObject) {
return null;
}
return $this->childProperty->getValue($embeddedObject);
}
public function setValue($object, $value)
/**
* {@inheritDoc}
*/
public function setValue($object, $value = null)
{
$embeddedObject = $this->parentProperty->getValue($object);
if ($embeddedObject === null) {
$embeddedObject = unserialize(sprintf('O:%d:"%s":0:{}', strlen($this->class), $this->class));
if (null === $embeddedObject) {
$this->instantiator = $this->instantiator ?: new Instantiator();
$embeddedObject = $this->instantiator->instantiate($this->class);
$this->parentProperty->setValue($object, $embeddedObject);
}

View File

@ -0,0 +1,15 @@
<?php
namespace Doctrine\Tests\Models\Reflection;
use ArrayObject;
/**
* A test asset extending {@see \ArrayObject}, useful for verifying internal classes issues with reflection
*/
class ArrayObjectExtendingClass extends ArrayObject
{
private $privateProperty;
protected $protectedProperty;
public $publicProperty;
}

View File

@ -0,0 +1,135 @@
<?php
namespace Doctrine\Tests\ORM\Mapping;
use Doctrine\Instantiator\Instantiator;
use Doctrine\ORM\Mapping\ReflectionEmbeddedProperty;
use ReflectionProperty;
/**
* Tests for {@see \Doctrine\ORM\Mapping\ReflectionEmbeddedProperty}
*
* @covers \Doctrine\ORM\Mapping\ReflectionEmbeddedProperty
*/
class ReflectionEmbeddedPropertyTest extends \PHPUnit_Framework_TestCase
{
/**
* @param ReflectionProperty $parentProperty
* @param ReflectionProperty $childProperty
*
* @dataProvider getTestedReflectionProperties
*/
public function testCanSetAndGetEmbeddedProperty(
ReflectionProperty $parentProperty,
ReflectionProperty $childProperty
) {
$embeddedPropertyReflection = new ReflectionEmbeddedProperty(
$parentProperty,
$childProperty,
$childProperty->getDeclaringClass()->getName()
);
$instantiator = new Instantiator();
$object = $instantiator->instantiate($parentProperty->getDeclaringClass()->getName());
$embeddedPropertyReflection->setValue($object, 'newValue');
$this->assertSame('newValue', $embeddedPropertyReflection->getValue($object));
$embeddedPropertyReflection->setValue($object, 'changedValue');
$this->assertSame('changedValue', $embeddedPropertyReflection->getValue($object));
}
/**
* @param ReflectionProperty $parentProperty
* @param ReflectionProperty $childProperty
*
* @dataProvider getTestedReflectionProperties
*/
public function testWillSkipReadingPropertiesFromNullEmbeddable(
ReflectionProperty $parentProperty,
ReflectionProperty $childProperty
)
{
$embeddedPropertyReflection = new ReflectionEmbeddedProperty(
$parentProperty,
$childProperty,
$childProperty->getDeclaringClass()->getName()
);
$instantiator = new Instantiator();
$this->assertNull($embeddedPropertyReflection->getValue(
$instantiator->instantiate($parentProperty->getDeclaringClass()->getName())
));
}
/**
* Data provider
*
* @return ReflectionProperty[][]
*/
public function getTestedReflectionProperties()
{
return array(
array(
$this->getReflectionProperty(
'Doctrine\\Tests\\Models\\Generic\\BooleanModel',
'id'
),
$this->getReflectionProperty(
'Doctrine\\Tests\\Models\\Generic\\BooleanModel',
'id'
),
),
// reflection on classes extending internal PHP classes:
array(
$this->getReflectionProperty(
'Doctrine\\Tests\\Models\\Reflection\\ArrayObjectExtendingClass',
'publicProperty'
),
$this->getReflectionProperty(
'Doctrine\\Tests\\Models\\Reflection\\ArrayObjectExtendingClass',
'privateProperty'
),
),
array(
$this->getReflectionProperty(
'Doctrine\\Tests\\Models\\Reflection\\ArrayObjectExtendingClass',
'publicProperty'
),
$this->getReflectionProperty(
'Doctrine\\Tests\\Models\\Reflection\\ArrayObjectExtendingClass',
'protectedProperty'
),
),
array(
$this->getReflectionProperty(
'Doctrine\\Tests\\Models\\Reflection\\ArrayObjectExtendingClass',
'publicProperty'
),
$this->getReflectionProperty(
'Doctrine\\Tests\\Models\\Reflection\\ArrayObjectExtendingClass',
'publicProperty'
),
),
);
}
/**
* @param string $className
* @param string $propertyName
*
* @return ReflectionProperty
*/
private function getReflectionProperty($className, $propertyName)
{
$reflectionProperty = new ReflectionProperty($className, $propertyName);
$reflectionProperty->setAccessible(true);
return $reflectionProperty;
}
}