348 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			348 lines
		
	
	
		
			12 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;
 | |
| 
 | |
| use Symfony\Component\VarExporter\Hydrator as PublicHydrator;
 | |
| use Symfony\Component\VarExporter\Internal\Hydrator;
 | |
| use Symfony\Component\VarExporter\Internal\LazyObjectRegistry as Registry;
 | |
| use Symfony\Component\VarExporter\Internal\LazyObjectState;
 | |
| use Symfony\Component\VarExporter\Internal\LazyObjectTrait;
 | |
| 
 | |
| trait LazyProxyTrait
 | |
| {
 | |
|     use LazyObjectTrait;
 | |
| 
 | |
|     /**
 | |
|      * Creates a lazy-loading virtual proxy.
 | |
|      *
 | |
|      * @param \Closure():object $initializer Returns the proxied object
 | |
|      * @param static|null       $instance
 | |
|      */
 | |
|     public static function createLazyProxy(\Closure $initializer, object $instance = null): static
 | |
|     {
 | |
|         if (self::class !== $class = $instance ? $instance::class : static::class) {
 | |
|             $skippedProperties = ["\0".self::class."\0lazyObjectState" => true];
 | |
|         } elseif (\defined($class.'::LAZY_OBJECT_PROPERTY_SCOPES')) {
 | |
|             Hydrator::$propertyScopes[$class] ??= $class::LAZY_OBJECT_PROPERTY_SCOPES;
 | |
|         }
 | |
| 
 | |
|         $instance ??= (Registry::$classReflectors[$class] ??= new \ReflectionClass($class))->newInstanceWithoutConstructor();
 | |
|         $instance->lazyObjectState = new LazyObjectState($initializer);
 | |
| 
 | |
|         foreach (Registry::$classResetters[$class] ??= Registry::getClassResetters($class) as $reset) {
 | |
|             $reset($instance, $skippedProperties ??= []);
 | |
|         }
 | |
| 
 | |
|         return $instance;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Returns whether the object is initialized.
 | |
|      *
 | |
|      * @param $partial Whether partially initialized objects should be considered as initialized
 | |
|      */
 | |
|     public function isLazyObjectInitialized(bool $partial = false): bool
 | |
|     {
 | |
|         return !isset($this->lazyObjectState) || isset($this->lazyObjectState->realInstance) || Registry::$noInitializerState === $this->lazyObjectState->initializer;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * Forces initialization of a lazy object and returns it.
 | |
|      */
 | |
|     public function initializeLazyObject(): parent
 | |
|     {
 | |
|         if ($state = $this->lazyObjectState ?? null) {
 | |
|             return $state->realInstance ??= ($state->initializer)();
 | |
|         }
 | |
| 
 | |
|         return $this;
 | |
|     }
 | |
| 
 | |
|     /**
 | |
|      * @return bool Returns false when the object cannot be reset, ie when it's not a lazy object
 | |
|      */
 | |
|     public function resetLazyObject(): bool
 | |
|     {
 | |
|         if (!isset($this->lazyObjectState) || Registry::$noInitializerState === $this->lazyObjectState->initializer) {
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         unset($this->lazyObjectState->realInstance);
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     public function &__get($name): mixed
 | |
|     {
 | |
|         $propertyScopes = Hydrator::$propertyScopes[$this::class] ??= Hydrator::getPropertyScopes($this::class);
 | |
|         $scope = null;
 | |
|         $instance = $this;
 | |
| 
 | |
|         if ([$class, , $readonlyScope] = $propertyScopes[$name] ?? null) {
 | |
|             $scope = Registry::getScope($propertyScopes, $class, $name);
 | |
| 
 | |
|             if (null === $scope || isset($propertyScopes["\0$scope\0$name"])) {
 | |
|                 if ($state = $this->lazyObjectState ?? null) {
 | |
|                     $instance = $state->realInstance ??= ($state->initializer)();
 | |
|                 }
 | |
|                 $parent = 2;
 | |
|                 goto get_in_scope;
 | |
|             }
 | |
|         }
 | |
|         $parent = (Registry::$parentMethods[self::class] ??= Registry::getParentMethods(self::class))['get'];
 | |
| 
 | |
|         if ($state = $this->lazyObjectState ?? null) {
 | |
|             $instance = $state->realInstance ??= ($state->initializer)();
 | |
|         } else {
 | |
|             if (2 === $parent) {
 | |
|                 return parent::__get($name);
 | |
|             }
 | |
|             $value = parent::__get($name);
 | |
| 
 | |
|             return $value;
 | |
|         }
 | |
| 
 | |
|         if (!$parent && null === $class && !\array_key_exists($name, (array) $instance)) {
 | |
|             $frame = debug_backtrace(\DEBUG_BACKTRACE_IGNORE_ARGS, 1)[0];
 | |
|             trigger_error(sprintf('Undefined property: %s::$%s in %s on line %s', $instance::class, $name, $frame['file'], $frame['line']), \E_USER_NOTICE);
 | |
|         }
 | |
| 
 | |
|         get_in_scope:
 | |
| 
 | |
|         try {
 | |
|             if (null === $scope) {
 | |
|                 if (null === $readonlyScope && 1 !== $parent) {
 | |
|                     return $instance->$name;
 | |
|                 }
 | |
|                 $value = $instance->$name;
 | |
| 
 | |
|                 return $value;
 | |
|             }
 | |
|             $accessor = Registry::$classAccessors[$scope] ??= Registry::getClassAccessors($scope);
 | |
| 
 | |
|             return $accessor['get']($instance, $name, null !== $readonlyScope || 1 === $parent);
 | |
|         } catch (\Error $e) {
 | |
|             if (\Error::class !== $e::class || !str_starts_with($e->getMessage(), 'Cannot access uninitialized non-nullable property')) {
 | |
|                 throw $e;
 | |
|             }
 | |
| 
 | |
|             try {
 | |
|                 if (null === $scope) {
 | |
|                     $instance->$name = [];
 | |
| 
 | |
|                     return $instance->$name;
 | |
|                 }
 | |
| 
 | |
|                 $accessor['set']($instance, $name, []);
 | |
| 
 | |
|                 return $accessor['get']($instance, $name, null !== $readonlyScope || 1 === $parent);
 | |
|             } catch (\Error) {
 | |
|                 throw $e;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public function __set($name, $value): void
 | |
|     {
 | |
|         $propertyScopes = Hydrator::$propertyScopes[$this::class] ??= Hydrator::getPropertyScopes($this::class);
 | |
|         $scope = null;
 | |
|         $instance = $this;
 | |
| 
 | |
|         if ([$class, , $readonlyScope] = $propertyScopes[$name] ?? null) {
 | |
|             $scope = Registry::getScope($propertyScopes, $class, $name, $readonlyScope);
 | |
| 
 | |
|             if ($readonlyScope === $scope || isset($propertyScopes["\0$scope\0$name"])) {
 | |
|                 if ($state = $this->lazyObjectState ?? null) {
 | |
|                     $instance = $state->realInstance ??= ($state->initializer)();
 | |
|                 }
 | |
|                 goto set_in_scope;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if ($state = $this->lazyObjectState ?? null) {
 | |
|             $instance = $state->realInstance ??= ($state->initializer)();
 | |
|         } elseif ((Registry::$parentMethods[self::class] ??= Registry::getParentMethods(self::class))['set']) {
 | |
|             parent::__set($name, $value);
 | |
| 
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         set_in_scope:
 | |
| 
 | |
|         if (null === $scope) {
 | |
|             $instance->$name = $value;
 | |
|         } else {
 | |
|             $accessor = Registry::$classAccessors[$scope] ??= Registry::getClassAccessors($scope);
 | |
|             $accessor['set']($instance, $name, $value);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public function __isset($name): bool
 | |
|     {
 | |
|         $propertyScopes = Hydrator::$propertyScopes[$this::class] ??= Hydrator::getPropertyScopes($this::class);
 | |
|         $scope = null;
 | |
|         $instance = $this;
 | |
| 
 | |
|         if ([$class] = $propertyScopes[$name] ?? null) {
 | |
|             $scope = Registry::getScope($propertyScopes, $class, $name);
 | |
| 
 | |
|             if (null === $scope || isset($propertyScopes["\0$scope\0$name"])) {
 | |
|                 if ($state = $this->lazyObjectState ?? null) {
 | |
|                     $instance = $state->realInstance ??= ($state->initializer)();
 | |
|                 }
 | |
|                 goto isset_in_scope;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if ($state = $this->lazyObjectState ?? null) {
 | |
|             $instance = $state->realInstance ??= ($state->initializer)();
 | |
|         } elseif ((Registry::$parentMethods[self::class] ??= Registry::getParentMethods(self::class))['isset']) {
 | |
|             return parent::__isset($name);
 | |
|         }
 | |
| 
 | |
|         isset_in_scope:
 | |
| 
 | |
|         if (null === $scope) {
 | |
|             return isset($instance->$name);
 | |
|         }
 | |
|         $accessor = Registry::$classAccessors[$scope] ??= Registry::getClassAccessors($scope);
 | |
| 
 | |
|         return $accessor['isset']($instance, $name);
 | |
|     }
 | |
| 
 | |
|     public function __unset($name): void
 | |
|     {
 | |
|         $propertyScopes = Hydrator::$propertyScopes[$this::class] ??= Hydrator::getPropertyScopes($this::class);
 | |
|         $scope = null;
 | |
|         $instance = $this;
 | |
| 
 | |
|         if ([$class, , $readonlyScope] = $propertyScopes[$name] ?? null) {
 | |
|             $scope = Registry::getScope($propertyScopes, $class, $name, $readonlyScope);
 | |
| 
 | |
|             if ($readonlyScope === $scope || isset($propertyScopes["\0$scope\0$name"])) {
 | |
|                 if ($state = $this->lazyObjectState ?? null) {
 | |
|                     $instance = $state->realInstance ??= ($state->initializer)();
 | |
|                 }
 | |
|                 goto unset_in_scope;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if ($state = $this->lazyObjectState ?? null) {
 | |
|             $instance = $state->realInstance ??= ($state->initializer)();
 | |
|         } elseif ((Registry::$parentMethods[self::class] ??= Registry::getParentMethods(self::class))['unset']) {
 | |
|             parent::__unset($name);
 | |
| 
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         unset_in_scope:
 | |
| 
 | |
|         if (null === $scope) {
 | |
|             unset($instance->$name);
 | |
|         } else {
 | |
|             $accessor = Registry::$classAccessors[$scope] ??= Registry::getClassAccessors($scope);
 | |
|             $accessor['unset']($instance, $name);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public function __clone(): void
 | |
|     {
 | |
|         if (!isset($this->lazyObjectState)) {
 | |
|             if ((Registry::$parentMethods[self::class] ??= Registry::getParentMethods(self::class))['clone']) {
 | |
|                 parent::__clone();
 | |
|             }
 | |
| 
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         $this->lazyObjectState = clone $this->lazyObjectState;
 | |
| 
 | |
|         if (isset($this->lazyObjectState->realInstance)) {
 | |
|             $this->lazyObjectState->realInstance = clone $this->lazyObjectState->realInstance;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public function __serialize(): array
 | |
|     {
 | |
|         $class = self::class;
 | |
|         $state = $this->lazyObjectState ?? null;
 | |
| 
 | |
|         if (!$state && (Registry::$parentMethods[$class] ??= Registry::getParentMethods($class))['serialize']) {
 | |
|             $properties = parent::__serialize();
 | |
|         } else {
 | |
|             $properties = (array) $this;
 | |
| 
 | |
|             if ($state) {
 | |
|                 unset($properties["\0$class\0lazyObjectState"]);
 | |
|                 $properties["\0$class\0lazyObjectReal"] = $state->realInstance ??= ($state->initializer)();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         if ($state || Registry::$parentMethods[$class]['serialize'] || !Registry::$parentMethods[$class]['sleep']) {
 | |
|             return $properties;
 | |
|         }
 | |
| 
 | |
|         $scope = get_parent_class($class);
 | |
|         $data = [];
 | |
| 
 | |
|         foreach (parent::__sleep() as $name) {
 | |
|             $value = $properties[$k = $name] ?? $properties[$k = "\0*\0$name"] ?? $properties[$k = "\0$scope\0$name"] ?? $k = null;
 | |
| 
 | |
|             if (null === $k) {
 | |
|                 trigger_error(sprintf('serialize(): "%s" returned as member variable from __sleep() but does not exist', $name), \E_USER_NOTICE);
 | |
|             } else {
 | |
|                 $data[$k] = $value;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return $data;
 | |
|     }
 | |
| 
 | |
|     public function __unserialize(array $data): void
 | |
|     {
 | |
|         $class = self::class;
 | |
| 
 | |
|         if ($instance = $data["\0$class\0lazyObjectReal"] ?? null) {
 | |
|             unset($data["\0$class\0lazyObjectReal"]);
 | |
| 
 | |
|             foreach (Registry::$classResetters[$class] ??= Registry::getClassResetters($class) as $reset) {
 | |
|                 $reset($this, $data);
 | |
|             }
 | |
| 
 | |
|             if ($data) {
 | |
|                 PublicHydrator::hydrate($this, $data);
 | |
|             }
 | |
|             $this->lazyObjectState = new LazyObjectState(Registry::$noInitializerState ??= static fn () => throw new \LogicException('Lazy proxy has no initializer.'));
 | |
|             $this->lazyObjectState->realInstance = $instance;
 | |
|         } elseif ((Registry::$parentMethods[$class] ??= Registry::getParentMethods($class))['unserialize']) {
 | |
|             parent::__unserialize($data);
 | |
|         } else {
 | |
|             PublicHydrator::hydrate($this, $data);
 | |
| 
 | |
|             if (Registry::$parentMethods[$class]['wakeup']) {
 | |
|                 parent::__wakeup();
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     public function __destruct()
 | |
|     {
 | |
|         if (isset($this->lazyObjectState)) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         if ((Registry::$parentMethods[self::class] ??= Registry::getParentMethods(self::class))['destruct']) {
 | |
|             parent::__destruct();
 | |
|         }
 | |
|     }
 | |
| }
 |