147 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			PHP
		
	
	
		
		
			
		
	
	
			147 lines
		
	
	
		
			5.6 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\VarExporter\Internal;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								use Symfony\Component\VarExporter\Exception\ClassNotFoundException;
							 | 
						||
| 
								 | 
							
								use Symfony\Component\VarExporter\Exception\NotInstantiableTypeException;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								/**
							 | 
						||
| 
								 | 
							
								 * @author Nicolas Grekas <p@tchwork.com>
							 | 
						||
| 
								 | 
							
								 *
							 | 
						||
| 
								 | 
							
								 * @internal
							 | 
						||
| 
								 | 
							
								 */
							 | 
						||
| 
								 | 
							
								class Registry
							 | 
						||
| 
								 | 
							
								{
							 | 
						||
| 
								 | 
							
								    public static $reflectors = [];
							 | 
						||
| 
								 | 
							
								    public static $prototypes = [];
							 | 
						||
| 
								 | 
							
								    public static $factories = [];
							 | 
						||
| 
								 | 
							
								    public static $cloneable = [];
							 | 
						||
| 
								 | 
							
								    public static $instantiableWithoutConstructor = [];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public function __construct(array $classes)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        foreach ($classes as $i => $class) {
							 | 
						||
| 
								 | 
							
								            $this->$i = $class;
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public static function unserialize($objects, $serializables)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $unserializeCallback = ini_set('unserialize_callback_func', __CLASS__.'::getClassReflector');
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        try {
							 | 
						||
| 
								 | 
							
								            foreach ($serializables as $k => $v) {
							 | 
						||
| 
								 | 
							
								                $objects[$k] = unserialize($v);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        } finally {
							 | 
						||
| 
								 | 
							
								            ini_set('unserialize_callback_func', $unserializeCallback);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return $objects;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public static function p($class)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        self::getClassReflector($class, true, true);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return self::$prototypes[$class];
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public static function f($class)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        $reflector = self::$reflectors[$class] ?? self::getClassReflector($class, true, false);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return self::$factories[$class] = \Closure::fromCallable([$reflector, 'newInstanceWithoutConstructor']);
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								    public static function getClassReflector($class, $instantiableWithoutConstructor = false, $cloneable = null)
							 | 
						||
| 
								 | 
							
								    {
							 | 
						||
| 
								 | 
							
								        if (!($isClass = class_exists($class)) && !interface_exists($class, false) && !trait_exists($class, false)) {
							 | 
						||
| 
								 | 
							
								            throw new ClassNotFoundException($class);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								        $reflector = new \ReflectionClass($class);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($instantiableWithoutConstructor) {
							 | 
						||
| 
								 | 
							
								            $proto = $reflector->newInstanceWithoutConstructor();
							 | 
						||
| 
								 | 
							
								        } elseif (!$isClass || $reflector->isAbstract()) {
							 | 
						||
| 
								 | 
							
								            throw new NotInstantiableTypeException($class);
							 | 
						||
| 
								 | 
							
								        } elseif ($reflector->name !== $class) {
							 | 
						||
| 
								 | 
							
								            $reflector = self::$reflectors[$name = $reflector->name] ?? self::getClassReflector($name, false, $cloneable);
							 | 
						||
| 
								 | 
							
								            self::$cloneable[$class] = self::$cloneable[$name];
							 | 
						||
| 
								 | 
							
								            self::$instantiableWithoutConstructor[$class] = self::$instantiableWithoutConstructor[$name];
							 | 
						||
| 
								 | 
							
								            self::$prototypes[$class] = self::$prototypes[$name];
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            return self::$reflectors[$class] = $reflector;
							 | 
						||
| 
								 | 
							
								        } else {
							 | 
						||
| 
								 | 
							
								            try {
							 | 
						||
| 
								 | 
							
								                $proto = $reflector->newInstanceWithoutConstructor();
							 | 
						||
| 
								 | 
							
								                $instantiableWithoutConstructor = true;
							 | 
						||
| 
								 | 
							
								            } catch (\ReflectionException $e) {
							 | 
						||
| 
								 | 
							
								                $proto = $reflector->implementsInterface('Serializable') && !method_exists($class, '__unserialize') ? 'C:' : 'O:';
							 | 
						||
| 
								 | 
							
								                if ('C:' === $proto && !$reflector->getMethod('unserialize')->isInternal()) {
							 | 
						||
| 
								 | 
							
								                    $proto = null;
							 | 
						||
| 
								 | 
							
								                } else {
							 | 
						||
| 
								 | 
							
								                    try {
							 | 
						||
| 
								 | 
							
								                        $proto = @unserialize($proto.\strlen($class).':"'.$class.'":0:{}');
							 | 
						||
| 
								 | 
							
								                    } catch (\Exception $e) {
							 | 
						||
| 
								 | 
							
								                        if (__FILE__ !== $e->getFile()) {
							 | 
						||
| 
								 | 
							
								                            throw $e;
							 | 
						||
| 
								 | 
							
								                        }
							 | 
						||
| 
								 | 
							
								                        throw new NotInstantiableTypeException($class, $e);
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                    if (false === $proto) {
							 | 
						||
| 
								 | 
							
								                        throw new NotInstantiableTypeException($class);
							 | 
						||
| 
								 | 
							
								                    }
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								            if (null !== $proto && !$proto instanceof \Throwable && !$proto instanceof \Serializable && !method_exists($class, '__sleep') && (\PHP_VERSION_ID < 70400 || !method_exists($class, '__serialize'))) {
							 | 
						||
| 
								 | 
							
								                try {
							 | 
						||
| 
								 | 
							
								                    serialize($proto);
							 | 
						||
| 
								 | 
							
								                } catch (\Exception $e) {
							 | 
						||
| 
								 | 
							
								                    throw new NotInstantiableTypeException($class, $e);
							 | 
						||
| 
								 | 
							
								                }
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if (null === $cloneable) {
							 | 
						||
| 
								 | 
							
								            if (($proto instanceof \Reflector || $proto instanceof \ReflectionGenerator || $proto instanceof \ReflectionType || $proto instanceof \IteratorIterator || $proto instanceof \RecursiveIteratorIterator) && (!$proto instanceof \Serializable && !method_exists($proto, '__wakeup') && (\PHP_VERSION_ID < 70400 || !method_exists($class, '__unserialize')))) {
							 | 
						||
| 
								 | 
							
								                throw new NotInstantiableTypeException($class);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $cloneable = $reflector->isCloneable() && !$reflector->hasMethod('__clone');
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        self::$cloneable[$class] = $cloneable;
							 | 
						||
| 
								 | 
							
								        self::$instantiableWithoutConstructor[$class] = $instantiableWithoutConstructor;
							 | 
						||
| 
								 | 
							
								        self::$prototypes[$class] = $proto;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        if ($proto instanceof \Throwable) {
							 | 
						||
| 
								 | 
							
								            static $setTrace;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            if (null === $setTrace) {
							 | 
						||
| 
								 | 
							
								                $setTrace = [
							 | 
						||
| 
								 | 
							
								                    new \ReflectionProperty(\Error::class, 'trace'),
							 | 
						||
| 
								 | 
							
								                    new \ReflectionProperty(\Exception::class, 'trace'),
							 | 
						||
| 
								 | 
							
								                ];
							 | 
						||
| 
								 | 
							
								                $setTrace[0]->setAccessible(true);
							 | 
						||
| 
								 | 
							
								                $setTrace[1]->setAccessible(true);
							 | 
						||
| 
								 | 
							
								                $setTrace[0] = \Closure::fromCallable([$setTrace[0], 'setValue']);
							 | 
						||
| 
								 | 
							
								                $setTrace[1] = \Closure::fromCallable([$setTrace[1], 'setValue']);
							 | 
						||
| 
								 | 
							
								            }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								            $setTrace[$proto instanceof \Exception]($proto, []);
							 | 
						||
| 
								 | 
							
								        }
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								        return self::$reflectors[$class] = $reflector;
							 | 
						||
| 
								 | 
							
								    }
							 | 
						||
| 
								 | 
							
								}
							 |