85 lines
2.8 KiB
PHP
85 lines
2.8 KiB
PHP
|
<?php
|
||
|
|
||
|
/*
|
||
|
* This file is part of the Symfony package.
|
||
|
*
|
||
|
* (c) Fabien Potencier <fabien@symfony.com>
|
||
|
*
|
||
|
* For the full copyright and license information, please view the LICENSE
|
||
|
* file that was distributed with this source code.
|
||
|
*/
|
||
|
|
||
|
namespace Symfony\Component\Validator\Mapping;
|
||
|
|
||
|
use Symfony\Component\Validator\Exception\ValidatorException;
|
||
|
|
||
|
/**
|
||
|
* Stores all metadata needed for validating a class property.
|
||
|
*
|
||
|
* The value of the property is obtained by directly accessing the property.
|
||
|
* The property will be accessed by reflection, so the access of private and
|
||
|
* protected properties is supported.
|
||
|
*
|
||
|
* This class supports serialization and cloning.
|
||
|
*
|
||
|
* @author Bernhard Schussek <bschussek@gmail.com>
|
||
|
*
|
||
|
* @see PropertyMetadataInterface
|
||
|
*/
|
||
|
class PropertyMetadata extends MemberMetadata
|
||
|
{
|
||
|
/**
|
||
|
* @param string $class The class this property is defined on
|
||
|
* @param string $name The name of this property
|
||
|
*
|
||
|
* @throws ValidatorException
|
||
|
*/
|
||
|
public function __construct(string $class, string $name)
|
||
|
{
|
||
|
if (!property_exists($class, $name)) {
|
||
|
throw new ValidatorException(sprintf('Property "%s" does not exist in class "%s".', $name, $class));
|
||
|
}
|
||
|
|
||
|
parent::__construct($class, $name, $name);
|
||
|
}
|
||
|
|
||
|
public function getPropertyValue(mixed $object): mixed
|
||
|
{
|
||
|
$reflProperty = $this->getReflectionMember($object);
|
||
|
|
||
|
if ($reflProperty->hasType() && !$reflProperty->isInitialized($object)) {
|
||
|
// There is no way to check if a property has been unset or if it is uninitialized.
|
||
|
// When trying to access an uninitialized property, __get method is triggered.
|
||
|
|
||
|
// If __get method is not present, no fallback is possible
|
||
|
// Otherwise we need to catch an Error in case we are trying to access an uninitialized but set property.
|
||
|
if (!method_exists($object, '__get')) {
|
||
|
return null;
|
||
|
}
|
||
|
|
||
|
try {
|
||
|
return $reflProperty->getValue($object);
|
||
|
} catch (\Error) {
|
||
|
return null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $reflProperty->getValue($object);
|
||
|
}
|
||
|
|
||
|
protected function newReflectionMember(object|string $objectOrClassName): \ReflectionMethod|\ReflectionProperty
|
||
|
{
|
||
|
$originalClass = \is_string($objectOrClassName) ? $objectOrClassName : $objectOrClassName::class;
|
||
|
|
||
|
while (!property_exists($objectOrClassName, $this->getName())) {
|
||
|
$objectOrClassName = get_parent_class($objectOrClassName);
|
||
|
|
||
|
if (false === $objectOrClassName) {
|
||
|
throw new ValidatorException(sprintf('Property "%s" does not exist in class "%s".', $this->getName(), $originalClass));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return new \ReflectionProperty($objectOrClassName, $this->getName());
|
||
|
}
|
||
|
}
|