| 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;    }}
 |