123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299 |
- <?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\VarExporter\Internal;
- use Symfony\Component\VarExporter\Exception\ClassNotFoundException;
- /**
- * @author Nicolas Grekas <p@tchwork.com>
- *
- * @internal
- */
- class Hydrator
- {
- public static $hydrators = [];
- public static $simpleHydrators = [];
- public static $propertyScopes = [];
- public $registry;
- public $values;
- public $properties;
- public $value;
- public $wakeups;
- public function __construct(?Registry $registry, ?Values $values, array $properties, $value, array $wakeups)
- {
- $this->registry = $registry;
- $this->values = $values;
- $this->properties = $properties;
- $this->value = $value;
- $this->wakeups = $wakeups;
- }
- public static function hydrate($objects, $values, $properties, $value, $wakeups)
- {
- foreach ($properties as $class => $vars) {
- (self::$hydrators[$class] ??= self::getHydrator($class))($vars, $objects);
- }
- foreach ($wakeups as $k => $v) {
- if (\is_array($v)) {
- $objects[-$k]->__unserialize($v);
- } else {
- $objects[$v]->__wakeup();
- }
- }
- return $value;
- }
- public static function getHydrator($class)
- {
- $baseHydrator = self::$hydrators['stdClass'] ??= static function ($properties, $objects) {
- foreach ($properties as $name => $values) {
- foreach ($values as $i => $v) {
- $objects[$i]->$name = $v;
- }
- }
- };
- switch ($class) {
- case 'stdClass':
- return $baseHydrator;
- case 'ErrorException':
- return $baseHydrator->bindTo(null, new class() extends \ErrorException {
- });
- case 'TypeError':
- return $baseHydrator->bindTo(null, new class() extends \Error {
- });
- case 'SplObjectStorage':
- return static function ($properties, $objects) {
- foreach ($properties as $name => $values) {
- if ("\0" === $name) {
- foreach ($values as $i => $v) {
- for ($j = 0; $j < \count($v); ++$j) {
- $objects[$i]->attach($v[$j], $v[++$j]);
- }
- }
- continue;
- }
- foreach ($values as $i => $v) {
- $objects[$i]->$name = $v;
- }
- }
- };
- }
- if (!class_exists($class) && !interface_exists($class, false) && !trait_exists($class, false)) {
- throw new ClassNotFoundException($class);
- }
- $classReflector = new \ReflectionClass($class);
- switch ($class) {
- case 'ArrayIterator':
- case 'ArrayObject':
- $constructor = $classReflector->getConstructor()->invokeArgs(...);
- return static function ($properties, $objects) use ($constructor) {
- foreach ($properties as $name => $values) {
- if ("\0" !== $name) {
- foreach ($values as $i => $v) {
- $objects[$i]->$name = $v;
- }
- }
- }
- foreach ($properties["\0"] ?? [] as $i => $v) {
- $constructor($objects[$i], $v);
- }
- };
- }
- if (!$classReflector->isInternal()) {
- return $baseHydrator->bindTo(null, $class);
- }
- if ($classReflector->name !== $class) {
- return self::$hydrators[$classReflector->name] ??= self::getHydrator($classReflector->name);
- }
- $propertySetters = [];
- foreach ($classReflector->getProperties() as $propertyReflector) {
- if (!$propertyReflector->isStatic()) {
- $propertySetters[$propertyReflector->name] = $propertyReflector->setValue(...);
- }
- }
- if (!$propertySetters) {
- return $baseHydrator;
- }
- return static function ($properties, $objects) use ($propertySetters) {
- foreach ($properties as $name => $values) {
- if ($setValue = $propertySetters[$name] ?? null) {
- foreach ($values as $i => $v) {
- $setValue($objects[$i], $v);
- }
- continue;
- }
- foreach ($values as $i => $v) {
- $objects[$i]->$name = $v;
- }
- }
- };
- }
- public static function getSimpleHydrator($class)
- {
- $baseHydrator = self::$simpleHydrators['stdClass'] ??= (function ($properties, $object) {
- $readonly = (array) $this;
- foreach ($properties as $name => &$value) {
- $object->$name = $value;
- if (!($readonly[$name] ?? false)) {
- $object->$name = &$value;
- }
- }
- })->bindTo(new \stdClass());
- switch ($class) {
- case 'stdClass':
- return $baseHydrator;
- case 'ErrorException':
- return $baseHydrator->bindTo(new \stdClass(), new class() extends \ErrorException {
- });
- case 'TypeError':
- return $baseHydrator->bindTo(new \stdClass(), new class() extends \Error {
- });
- case 'SplObjectStorage':
- return static function ($properties, $object) {
- foreach ($properties as $name => &$value) {
- if ("\0" !== $name) {
- $object->$name = $value;
- $object->$name = &$value;
- continue;
- }
- for ($i = 0; $i < \count($value); ++$i) {
- $object->attach($value[$i], $value[++$i]);
- }
- }
- };
- }
- if (!class_exists($class) && !interface_exists($class, false) && !trait_exists($class, false)) {
- throw new ClassNotFoundException($class);
- }
- $classReflector = new \ReflectionClass($class);
- switch ($class) {
- case 'ArrayIterator':
- case 'ArrayObject':
- $constructor = $classReflector->getConstructor()->invokeArgs(...);
- return static function ($properties, $object) use ($constructor) {
- foreach ($properties as $name => &$value) {
- if ("\0" === $name) {
- $constructor($object, $value);
- } else {
- $object->$name = $value;
- $object->$name = &$value;
- }
- }
- };
- }
- if (!$classReflector->isInternal()) {
- $readonly = new \stdClass();
- foreach ($classReflector->getProperties(\ReflectionProperty::IS_READONLY) as $propertyReflector) {
- if ($class === $propertyReflector->class) {
- $readonly->{$propertyReflector->name} = true;
- }
- }
- return $baseHydrator->bindTo($readonly, $class);
- }
- if ($classReflector->name !== $class) {
- return self::$simpleHydrators[$classReflector->name] ??= self::getSimpleHydrator($classReflector->name);
- }
- $propertySetters = [];
- foreach ($classReflector->getProperties() as $propertyReflector) {
- if (!$propertyReflector->isStatic()) {
- $propertySetters[$propertyReflector->name] = $propertyReflector->setValue(...);
- }
- }
- if (!$propertySetters) {
- return $baseHydrator;
- }
- return static function ($properties, $object) use ($propertySetters) {
- foreach ($properties as $name => &$value) {
- if ($setValue = $propertySetters[$name] ?? null) {
- $setValue($object, $value);
- } else {
- $object->$name = $value;
- $object->$name = &$value;
- }
- }
- };
- }
- /**
- * @return array
- */
- public static function getPropertyScopes($class)
- {
- $propertyScopes = [];
- $r = new \ReflectionClass($class);
- foreach ($r->getProperties() as $property) {
- $flags = $property->getModifiers();
- if (\ReflectionProperty::IS_STATIC & $flags) {
- continue;
- }
- $name = $property->name;
- if (\ReflectionProperty::IS_PRIVATE & $flags) {
- $propertyScopes["\0$class\0$name"] = $propertyScopes[$name] = [$class, $name, $flags & \ReflectionProperty::IS_READONLY ? $class : null];
- continue;
- }
- $propertyScopes[$name] = [$class, $name, $flags & \ReflectionProperty::IS_READONLY ? $property->class : null];
- if (\ReflectionProperty::IS_PROTECTED & $flags) {
- $propertyScopes["\0*\0$name"] = $propertyScopes[$name];
- }
- }
- while ($r = $r->getParentClass()) {
- $class = $r->name;
- foreach ($r->getProperties(\ReflectionProperty::IS_PRIVATE) as $property) {
- if (!$property->isStatic()) {
- $name = $property->name;
- $readonlyScope = $property->isReadOnly() ? $class : null;
- $propertyScopes["\0$class\0$name"] = [$class, $name, $readonlyScope];
- $propertyScopes[$name] ??= [$class, $name, $readonlyScope];
- }
- }
- }
- return $propertyScopes;
- }
- }
|