vendor/symfony/var-exporter/LazyGhostTrait.php line 175

  1. <?php
  2. /*
  3.  * This file is part of the Symfony package.
  4.  *
  5.  * (c) Fabien Potencier <fabien@symfony.com>
  6.  *
  7.  * For the full copyright and license information, please view the LICENSE
  8.  * file that was distributed with this source code.
  9.  */
  10. namespace Symfony\Component\VarExporter;
  11. use Symfony\Component\VarExporter\Internal\Hydrator;
  12. use Symfony\Component\VarExporter\Internal\LazyObjectRegistry as Registry;
  13. use Symfony\Component\VarExporter\Internal\LazyObjectState;
  14. trait LazyGhostTrait
  15. {
  16.     private LazyObjectState $lazyObjectState;
  17.     /**
  18.      * Creates a lazy-loading ghost instance.
  19.      *
  20.      * When the initializer is a closure, it should initialize all properties at
  21.      * once and is given the instance to initialize as argument.
  22.      *
  23.      * When the initializer is an array of closures, it should be indexed by
  24.      * properties and closures should accept 4 arguments: the instance to
  25.      * initialize, the property to initialize, its write-scope, and its default
  26.      * value. Each closure should return the value of the corresponding property.
  27.      * The special "\0" key can be used to define a closure that returns all
  28.      * properties at once when full-initialization is needed; it takes the
  29.      * instance and its default properties as arguments.
  30.      *
  31.      * Properties should be indexed by their array-cast name, see
  32.      * https://php.net/manual/language.types.array#language.types.array.casting
  33.      *
  34.      * @param (\Closure(static):void
  35.      *        |array<string, \Closure(static, string, ?string, mixed):mixed>
  36.      *        |array{"\0": \Closure(static, array<string, mixed>):array<string, mixed>}) $initializer
  37.      * @param array<string, true>|null $skippedProperties An array indexed by the properties to skip, aka the ones
  38.      *                                                    that the initializer doesn't set when its a closure
  39.      * @param static|null              $instance
  40.      */
  41.     public static function createLazyGhost(\Closure|array $initializer, array $skippedProperties nullobject $instance null): static
  42.     {
  43.         $onlyProperties null === $skippedProperties && \is_array($initializer) ? $initializer null;
  44.         if (self::class !== $class $instance $instance::class : static::class) {
  45.             $skippedProperties["\0".self::class."\0lazyObjectState"] = true;
  46.         } elseif (\defined($class.'::LAZY_OBJECT_PROPERTY_SCOPES')) {
  47.             Hydrator::$propertyScopes[$class] ??= $class::LAZY_OBJECT_PROPERTY_SCOPES;
  48.         }
  49.         $instance ??= (Registry::$classReflectors[$class] ??= new \ReflectionClass($class))->newInstanceWithoutConstructor();
  50.         Registry::$defaultProperties[$class] ??= (array) $instance;
  51.         $instance->lazyObjectState = new LazyObjectState($initializer$skippedProperties ??= []);
  52.         foreach (Registry::$classResetters[$class] ??= Registry::getClassResetters($class) as $reset) {
  53.             $reset($instance$skippedProperties$onlyProperties);
  54.         }
  55.         return $instance;
  56.     }
  57.     /**
  58.      * Returns whether the object is initialized.
  59.      *
  60.      * @param $partial Whether partially initialized objects should be considered as initialized
  61.      */
  62.     public function isLazyObjectInitialized(bool $partial false): bool
  63.     {
  64.         if (!$state $this->lazyObjectState ?? null) {
  65.             return true;
  66.         }
  67.         if (!\is_array($state->initializer)) {
  68.             return LazyObjectState::STATUS_INITIALIZED_FULL === $state->status;
  69.         }
  70.         $class $this::class;
  71.         $properties = (array) $this;
  72.         if ($partial) {
  73.             return (bool) array_intersect_key($state->initializer$properties);
  74.         }
  75.         $propertyScopes Hydrator::$propertyScopes[$class] ??= Hydrator::getPropertyScopes($class);
  76.         foreach ($state->initializer as $key => $initializer) {
  77.             if (!\array_key_exists($key$properties) && isset($propertyScopes[$key])) {
  78.                 return false;
  79.             }
  80.         }
  81.         return true;
  82.     }
  83.     /**
  84.      * Forces initialization of a lazy object and returns it.
  85.      */
  86.     public function initializeLazyObject(): static
  87.     {
  88.         if (!$state $this->lazyObjectState ?? null) {
  89.             return $this;
  90.         }
  91.         if (!\is_array($state->initializer)) {
  92.             if (LazyObjectState::STATUS_UNINITIALIZED_FULL === $state->status) {
  93.                 $state->initialize($this''null);
  94.             }
  95.             return $this;
  96.         }
  97.         $values = isset($state->initializer["\0"]) ? null : [];
  98.         $class $this::class;
  99.         $properties = (array) $this;
  100.         $propertyScopes Hydrator::$propertyScopes[$class] ??= Hydrator::getPropertyScopes($class);
  101.         foreach ($state->initializer as $key => $initializer) {
  102.             if (\array_key_exists($key$properties) || ![$scope$name$readonlyScope] = $propertyScopes[$key] ?? null) {
  103.                 continue;
  104.             }
  105.             $scope $readonlyScope ?? ('*' !== $scope $scope $class);
  106.             if (null === $values) {
  107.                 if (!\is_array($values = ($state->initializer["\0"])($thisRegistry::$defaultProperties[$class]))) {
  108.                     throw new \TypeError(sprintf('The lazy-initializer defined for instance of "%s" must return an array, got "%s".'$classget_debug_type($values)));
  109.                 }
  110.                 if (\array_key_exists($key$properties = (array) $this)) {
  111.                     continue;
  112.                 }
  113.             }
  114.             if (\array_key_exists($key$values)) {
  115.                 $accessor Registry::$classAccessors[$scope] ??= Registry::getClassAccessors($scope);
  116.                 $accessor['set']($this$name$properties[$key] = $values[$key]);
  117.             } else {
  118.                 $state->initialize($this$name$scope);
  119.                 $properties = (array) $this;
  120.             }
  121.         }
  122.         return $this;
  123.     }
  124.     /**
  125.      * @return bool Returns false when the object cannot be reset, ie when it's not a lazy object
  126.      */
  127.     public function resetLazyObject(): bool
  128.     {
  129.         if (!$state $this->lazyObjectState ?? null) {
  130.             return false;
  131.         }
  132.         if (LazyObjectState::STATUS_UNINITIALIZED_FULL !== $state->status) {
  133.             $state->reset($this);
  134.         }
  135.         return true;
  136.     }
  137.     public function &__get($name): mixed
  138.     {
  139.         $propertyScopes Hydrator::$propertyScopes[$this::class] ??= Hydrator::getPropertyScopes($this::class);
  140.         $scope null;
  141.         if ([$class, , $readonlyScope] = $propertyScopes[$name] ?? null) {
  142.             $scope Registry::getScope($propertyScopes$class$name);
  143.             $state $this->lazyObjectState ?? null;
  144.             if ($state && (null === $scope || isset($propertyScopes["\0$scope\0$name"]))
  145.                 && LazyObjectState::STATUS_UNINITIALIZED_PARTIAL !== $state->initialize($this$name$readonlyScope ?? $scope)
  146.             ) {
  147.                 goto get_in_scope;
  148.             }
  149.         }
  150.         if ($parent = (Registry::$parentMethods[self::class] ??= Registry::getParentMethods(self::class))['get']) {
  151.             if (=== $parent) {
  152.                 return parent::__get($name);
  153.             }
  154.             $value parent::__get($name);
  155.             return $value;
  156.         }
  157.         if (null === $class) {
  158.             $frame debug_backtrace(\DEBUG_BACKTRACE_IGNORE_ARGS1)[0];
  159.             trigger_error(sprintf('Undefined property: %s::$%s in %s on line %s'$this::class, $name$frame['file'], $frame['line']), \E_USER_NOTICE);
  160.         }
  161.         get_in_scope:
  162.         try {
  163.             if (null === $scope) {
  164.                 if (null === $readonlyScope) {
  165.                     return $this->$name;
  166.                 }
  167.                 $value $this->$name;
  168.                 return $value;
  169.             }
  170.             $accessor Registry::$classAccessors[$scope] ??= Registry::getClassAccessors($scope);
  171.             return $accessor['get']($this$namenull !== $readonlyScope);
  172.         } catch (\Error $e) {
  173.             if (\Error::class !== $e::class || !str_starts_with($e->getMessage(), 'Cannot access uninitialized non-nullable property')) {
  174.                 throw $e;
  175.             }
  176.             try {
  177.                 if (null === $scope) {
  178.                     $this->$name = [];
  179.                     return $this->$name;
  180.                 }
  181.                 $accessor['set']($this$name, []);
  182.                 return $accessor['get']($this$namenull !== $readonlyScope);
  183.             } catch (\Error) {
  184.                 throw $e;
  185.             }
  186.         }
  187.     }
  188.     public function __set($name$value): void
  189.     {
  190.         $propertyScopes Hydrator::$propertyScopes[$this::class] ??= Hydrator::getPropertyScopes($this::class);
  191.         $scope null;
  192.         $state null;
  193.         if ([$class, , $readonlyScope] = $propertyScopes[$name] ?? null) {
  194.             $scope Registry::getScope($propertyScopes$class$name$readonlyScope);
  195.             $state $this->lazyObjectState ?? null;
  196.             if ($state && ($readonlyScope === $scope || isset($propertyScopes["\0$scope\0$name"]))) {
  197.                 if (LazyObjectState::STATUS_UNINITIALIZED_FULL === $state->status) {
  198.                     $state->initialize($this$name$readonlyScope ?? $scope);
  199.                 }
  200.                 goto set_in_scope;
  201.             }
  202.         }
  203.         if ((Registry::$parentMethods[self::class] ??= Registry::getParentMethods(self::class))['set']) {
  204.             parent::__set($name$value);
  205.             return;
  206.         }
  207.         set_in_scope:
  208.         if (null === $scope) {
  209.             $this->$name $value;
  210.         } else {
  211.             $accessor Registry::$classAccessors[$scope] ??= Registry::getClassAccessors($scope);
  212.             $accessor['set']($this$name$value);
  213.         }
  214.     }
  215.     public function __isset($name): bool
  216.     {
  217.         $propertyScopes Hydrator::$propertyScopes[$this::class] ??= Hydrator::getPropertyScopes($this::class);
  218.         $scope null;
  219.         if ([$class, , $readonlyScope] = $propertyScopes[$name] ?? null) {
  220.             $scope Registry::getScope($propertyScopes$class$name);
  221.             $state $this->lazyObjectState ?? null;
  222.             if ($state && (null === $scope || isset($propertyScopes["\0$scope\0$name"]))
  223.                 && LazyObjectState::STATUS_UNINITIALIZED_PARTIAL !== $state->initialize($this$name$readonlyScope ?? $scope)
  224.             ) {
  225.                 goto isset_in_scope;
  226.             }
  227.         }
  228.         if ((Registry::$parentMethods[self::class] ??= Registry::getParentMethods(self::class))['isset']) {
  229.             return parent::__isset($name);
  230.         }
  231.         isset_in_scope:
  232.         if (null === $scope) {
  233.             return isset($this->$name);
  234.         }
  235.         $accessor Registry::$classAccessors[$scope] ??= Registry::getClassAccessors($scope);
  236.         return $accessor['isset']($this$name);
  237.     }
  238.     public function __unset($name): void
  239.     {
  240.         $propertyScopes Hydrator::$propertyScopes[$this::class] ??= Hydrator::getPropertyScopes($this::class);
  241.         $scope null;
  242.         if ([$class, , $readonlyScope] = $propertyScopes[$name] ?? null) {
  243.             $scope Registry::getScope($propertyScopes$class$name$readonlyScope);
  244.             $state $this->lazyObjectState ?? null;
  245.             if ($state && ($readonlyScope === $scope || isset($propertyScopes["\0$scope\0$name"]))) {
  246.                 if (LazyObjectState::STATUS_UNINITIALIZED_FULL === $state->status) {
  247.                     $state->initialize($this$name$readonlyScope ?? $scope);
  248.                 }
  249.                 goto unset_in_scope;
  250.             }
  251.         }
  252.         if ((Registry::$parentMethods[self::class] ??= Registry::getParentMethods(self::class))['unset']) {
  253.             parent::__unset($name);
  254.             return;
  255.         }
  256.         unset_in_scope:
  257.         if (null === $scope) {
  258.             unset($this->$name);
  259.         } else {
  260.             $accessor Registry::$classAccessors[$scope] ??= Registry::getClassAccessors($scope);
  261.             $accessor['unset']($this$name);
  262.         }
  263.     }
  264.     public function __clone(): void
  265.     {
  266.         if ($state $this->lazyObjectState ?? null) {
  267.             $this->lazyObjectState = clone $state;
  268.         }
  269.         if ((Registry::$parentMethods[self::class] ??= Registry::getParentMethods(self::class))['clone']) {
  270.             parent::__clone();
  271.         }
  272.     }
  273.     public function __serialize(): array
  274.     {
  275.         $class self::class;
  276.         if ((Registry::$parentMethods[$class] ??= Registry::getParentMethods($class))['serialize']) {
  277.             $properties parent::__serialize();
  278.         } else {
  279.             $this->initializeLazyObject();
  280.             $properties = (array) $this;
  281.         }
  282.         unset($properties["\0$class\0lazyObjectState"]);
  283.         if (Registry::$parentMethods[$class]['serialize'] || !Registry::$parentMethods[$class]['sleep']) {
  284.             return $properties;
  285.         }
  286.         $scope get_parent_class($class);
  287.         $data = [];
  288.         foreach (parent::__sleep() as $name) {
  289.             $value $properties[$k $name] ?? $properties[$k "\0*\0$name"] ?? $properties[$k "\0$scope\0$name"] ?? $k null;
  290.             if (null === $k) {
  291.                 trigger_error(sprintf('serialize(): "%s" returned as member variable from __sleep() but does not exist'$name), \E_USER_NOTICE);
  292.             } else {
  293.                 $data[$k] = $value;
  294.             }
  295.         }
  296.         return $data;
  297.     }
  298.     public function __destruct()
  299.     {
  300.         $state $this->lazyObjectState ?? null;
  301.         if ($state && \in_array($state->status, [LazyObjectState::STATUS_UNINITIALIZED_FULLLazyObjectState::STATUS_UNINITIALIZED_PARTIAL], true)) {
  302.             return;
  303.         }
  304.         if ((Registry::$parentMethods[self::class] ??= Registry::getParentMethods(self::class))['destruct']) {
  305.             parent::__destruct();
  306.         }
  307.     }
  308.     private function setLazyObjectAsInitialized(bool $initialized): void
  309.     {
  310.         $state $this->lazyObjectState ?? null;
  311.         if ($state && !\is_array($state->initializer)) {
  312.             $state->status $initialized LazyObjectState::STATUS_INITIALIZED_FULL LazyObjectState::STATUS_UNINITIALIZED_FULL;
  313.         }
  314.     }
  315. }