vendor/doctrine/common/lib/Doctrine/Common/Proxy/ProxyGenerator.php line 1159

Open in your IDE?
  1. <?php
  2. namespace Doctrine\Common\Proxy;
  3. use BackedEnum;
  4. use Doctrine\Common\Proxy\Exception\InvalidArgumentException;
  5. use Doctrine\Common\Proxy\Exception\UnexpectedValueException;
  6. use Doctrine\Common\Util\ClassUtils;
  7. use Doctrine\Persistence\Mapping\ClassMetadata;
  8. use ReflectionIntersectionType;
  9. use ReflectionMethod;
  10. use ReflectionNamedType;
  11. use ReflectionParameter;
  12. use ReflectionProperty;
  13. use ReflectionType;
  14. use ReflectionUnionType;
  15. use function array_combine;
  16. use function array_diff;
  17. use function array_key_exists;
  18. use function array_map;
  19. use function array_slice;
  20. use function array_unique;
  21. use function assert;
  22. use function call_user_func;
  23. use function chmod;
  24. use function class_exists;
  25. use function dirname;
  26. use function explode;
  27. use function file;
  28. use function file_put_contents;
  29. use function get_class;
  30. use function implode;
  31. use function in_array;
  32. use function interface_exists;
  33. use function is_callable;
  34. use function is_dir;
  35. use function is_string;
  36. use function is_writable;
  37. use function lcfirst;
  38. use function ltrim;
  39. use function method_exists;
  40. use function mkdir;
  41. use function preg_match;
  42. use function preg_match_all;
  43. use function rename;
  44. use function rtrim;
  45. use function sprintf;
  46. use function str_replace;
  47. use function strrev;
  48. use function strtolower;
  49. use function strtr;
  50. use function substr;
  51. use function trim;
  52. use function uniqid;
  53. use function var_export;
  54. use const DIRECTORY_SEPARATOR;
  55. use const PHP_VERSION_ID;
  56. /**
  57.  * This factory is used to generate proxy classes.
  58.  * It builds proxies from given parameters, a template and class metadata.
  59.  */
  60. class ProxyGenerator
  61. {
  62.     /**
  63.      * Used to match very simple id methods that don't need
  64.      * to be decorated since the identifier is known.
  65.      */
  66.     public const PATTERN_MATCH_ID_METHOD '((public\s+)?(function\s+%s\s*\(\)\s*)\s*(?::\s*\??\s*\\\\?[a-z_\x7f-\xff][\w\x7f-\xff]*(?:\\\\[a-z_\x7f-\xff][\w\x7f-\xff]*)*\s*)?{\s*return\s*\$this->%s;\s*})i';
  67.     /**
  68.      * The namespace that contains all proxy classes.
  69.      *
  70.      * @var string
  71.      */
  72.     private $proxyNamespace;
  73.     /**
  74.      * The directory that contains all proxy classes.
  75.      *
  76.      * @var string
  77.      */
  78.     private $proxyDirectory;
  79.     /**
  80.      * Map of callables used to fill in placeholders set in the template.
  81.      *
  82.      * @var string[]|callable[]
  83.      */
  84.     protected $placeholders = [
  85.         'baseProxyInterface'   => Proxy::class,
  86.         'additionalProperties' => '',
  87.     ];
  88.     /**
  89.      * Template used as a blueprint to generate proxies.
  90.      *
  91.      * @var string
  92.      */
  93.     protected $proxyClassTemplate '<?php
  94. namespace <namespace>;
  95. <enumUseStatements>
  96. /**
  97.  * DO NOT EDIT THIS FILE - IT WAS CREATED BY DOCTRINE\'S PROXY GENERATOR
  98.  */
  99. class <proxyShortClassName> extends \<className> implements \<baseProxyInterface>
  100. {
  101.     /**
  102.      * @var \Closure the callback responsible for loading properties in the proxy object. This callback is called with
  103.      *      three parameters, being respectively the proxy object to be initialized, the method that triggered the
  104.      *      initialization process and an array of ordered parameters that were passed to that method.
  105.      *
  106.      * @see \Doctrine\Common\Proxy\Proxy::__setInitializer
  107.      */
  108.     public $__initializer__;
  109.     /**
  110.      * @var \Closure the callback responsible of loading properties that need to be copied in the cloned object
  111.      *
  112.      * @see \Doctrine\Common\Proxy\Proxy::__setCloner
  113.      */
  114.     public $__cloner__;
  115.     /**
  116.      * @var boolean flag indicating if this object was already initialized
  117.      *
  118.      * @see \Doctrine\Persistence\Proxy::__isInitialized
  119.      */
  120.     public $__isInitialized__ = false;
  121.     /**
  122.      * @var array<string, null> properties to be lazy loaded, indexed by property name
  123.      */
  124.     public static $lazyPropertiesNames = <lazyPropertiesNames>;
  125.     /**
  126.      * @var array<string, mixed> default values of properties to be lazy loaded, with keys being the property names
  127.      *
  128.      * @see \Doctrine\Common\Proxy\Proxy::__getLazyProperties
  129.      */
  130.     public static $lazyPropertiesDefaults = <lazyPropertiesDefaults>;
  131. <additionalProperties>
  132. <constructorImpl>
  133. <magicGet>
  134. <magicSet>
  135. <magicIsset>
  136. <sleepImpl>
  137. <wakeupImpl>
  138. <cloneImpl>
  139.     /**
  140.      * Forces initialization of the proxy
  141.      */
  142.     public function __load()
  143.     {
  144.         $this->__initializer__ && $this->__initializer__->__invoke($this, \'__load\', []);
  145.     }
  146.     /**
  147.      * {@inheritDoc}
  148.      * @internal generated method: use only when explicitly handling proxy specific loading logic
  149.      */
  150.     public function __isInitialized()
  151.     {
  152.         return $this->__isInitialized__;
  153.     }
  154.     /**
  155.      * {@inheritDoc}
  156.      * @internal generated method: use only when explicitly handling proxy specific loading logic
  157.      */
  158.     public function __setInitialized($initialized)
  159.     {
  160.         $this->__isInitialized__ = $initialized;
  161.     }
  162.     /**
  163.      * {@inheritDoc}
  164.      * @internal generated method: use only when explicitly handling proxy specific loading logic
  165.      */
  166.     public function __setInitializer(\Closure $initializer = null)
  167.     {
  168.         $this->__initializer__ = $initializer;
  169.     }
  170.     /**
  171.      * {@inheritDoc}
  172.      * @internal generated method: use only when explicitly handling proxy specific loading logic
  173.      */
  174.     public function __getInitializer()
  175.     {
  176.         return $this->__initializer__;
  177.     }
  178.     /**
  179.      * {@inheritDoc}
  180.      * @internal generated method: use only when explicitly handling proxy specific loading logic
  181.      */
  182.     public function __setCloner(\Closure $cloner = null)
  183.     {
  184.         $this->__cloner__ = $cloner;
  185.     }
  186.     /**
  187.      * {@inheritDoc}
  188.      * @internal generated method: use only when explicitly handling proxy specific cloning logic
  189.      */
  190.     public function __getCloner()
  191.     {
  192.         return $this->__cloner__;
  193.     }
  194.     /**
  195.      * {@inheritDoc}
  196.      * @internal generated method: use only when explicitly handling proxy specific loading logic
  197.      * @deprecated no longer in use - generated code now relies on internal components rather than generated public API
  198.      * @static
  199.      */
  200.     public function __getLazyProperties()
  201.     {
  202.         return self::$lazyPropertiesDefaults;
  203.     }
  204.     <methods>
  205. }
  206. ';
  207.     /**
  208.      * Initializes a new instance of the <tt>ProxyFactory</tt> class that is
  209.      * connected to the given <tt>EntityManager</tt>.
  210.      *
  211.      * @param string $proxyDirectory The directory to use for the proxy classes. It must exist.
  212.      * @param string $proxyNamespace The namespace to use for the proxy classes.
  213.      *
  214.      * @throws InvalidArgumentException
  215.      */
  216.     public function __construct($proxyDirectory$proxyNamespace)
  217.     {
  218.         if (! $proxyDirectory) {
  219.             throw InvalidArgumentException::proxyDirectoryRequired();
  220.         }
  221.         if (! $proxyNamespace) {
  222.             throw InvalidArgumentException::proxyNamespaceRequired();
  223.         }
  224.         $this->proxyDirectory $proxyDirectory;
  225.         $this->proxyNamespace $proxyNamespace;
  226.     }
  227.     /**
  228.      * Sets a placeholder to be replaced in the template.
  229.      *
  230.      * @param string          $name
  231.      * @param string|callable $placeholder
  232.      *
  233.      * @throws InvalidArgumentException
  234.      */
  235.     public function setPlaceholder($name$placeholder)
  236.     {
  237.         if (! is_string($placeholder) && ! is_callable($placeholder)) {
  238.             throw InvalidArgumentException::invalidPlaceholder($name);
  239.         }
  240.         $this->placeholders[$name] = $placeholder;
  241.     }
  242.     /**
  243.      * Sets the base template used to create proxy classes.
  244.      *
  245.      * @param string $proxyClassTemplate
  246.      */
  247.     public function setProxyClassTemplate($proxyClassTemplate)
  248.     {
  249.         $this->proxyClassTemplate = (string) $proxyClassTemplate;
  250.     }
  251.     /**
  252.      * Generates a proxy class file.
  253.      *
  254.      * @param ClassMetadata $class    Metadata for the original class.
  255.      * @param string|bool   $fileName Filename (full path) for the generated class. If none is given, eval() is used.
  256.      *
  257.      * @throws InvalidArgumentException
  258.      * @throws UnexpectedValueException
  259.      */
  260.     public function generateProxyClass(ClassMetadata $class$fileName false)
  261.     {
  262.         $this->verifyClassCanBeProxied($class);
  263.         preg_match_all('(<([a-zA-Z]+)>)'$this->proxyClassTemplate$placeholderMatches);
  264.         $placeholderMatches array_combine($placeholderMatches[0], $placeholderMatches[1]);
  265.         $placeholders       = [];
  266.         foreach ($placeholderMatches as $placeholder => $name) {
  267.             $placeholders[$placeholder] = $this->placeholders[$name] ?? [$this'generate' $name];
  268.         }
  269.         foreach ($placeholders as & $placeholder) {
  270.             if (! is_callable($placeholder)) {
  271.                 continue;
  272.             }
  273.             $placeholder call_user_func($placeholder$class);
  274.         }
  275.         $proxyCode strtr($this->proxyClassTemplate$placeholders);
  276.         if (! $fileName) {
  277.             $proxyClassName $this->generateNamespace($class) . '\\' $this->generateProxyShortClassName($class);
  278.             if (! class_exists($proxyClassName)) {
  279.                 eval(substr($proxyCode5));
  280.             }
  281.             return;
  282.         }
  283.         $parentDirectory dirname($fileName);
  284.         if (! is_dir($parentDirectory) && (@mkdir($parentDirectory0775true) === false)) {
  285.             throw UnexpectedValueException::proxyDirectoryNotWritable($this->proxyDirectory);
  286.         }
  287.         if (! is_writable($parentDirectory)) {
  288.             throw UnexpectedValueException::proxyDirectoryNotWritable($this->proxyDirectory);
  289.         }
  290.         $tmpFileName $fileName '.' uniqid(''true);
  291.         file_put_contents($tmpFileName$proxyCode);
  292.         @chmod($tmpFileName0664);
  293.         rename($tmpFileName$fileName);
  294.     }
  295.     /**
  296.      * @throws InvalidArgumentException
  297.      */
  298.     private function verifyClassCanBeProxied(ClassMetadata $class)
  299.     {
  300.         if ($class->getReflectionClass()->isFinal()) {
  301.             throw InvalidArgumentException::classMustNotBeFinal($class->getName());
  302.         }
  303.         if ($class->getReflectionClass()->isAbstract()) {
  304.             throw InvalidArgumentException::classMustNotBeAbstract($class->getName());
  305.         }
  306.     }
  307.     /**
  308.      * Generates the proxy short class name to be used in the template.
  309.      *
  310.      * @return string
  311.      */
  312.     private function generateProxyShortClassName(ClassMetadata $class)
  313.     {
  314.         $proxyClassName ClassUtils::generateProxyClassName($class->getName(), $this->proxyNamespace);
  315.         $parts          explode('\\'strrev($proxyClassName), 2);
  316.         return strrev($parts[0]);
  317.     }
  318.     /**
  319.      * Generates the proxy namespace.
  320.      *
  321.      * @return string
  322.      */
  323.     private function generateNamespace(ClassMetadata $class)
  324.     {
  325.         $proxyClassName ClassUtils::generateProxyClassName($class->getName(), $this->proxyNamespace);
  326.         $parts          explode('\\'strrev($proxyClassName), 2);
  327.         return strrev($parts[1]);
  328.     }
  329.     /**
  330.      * Enums must have a use statement when used as public property defaults.
  331.      */
  332.     public function generateEnumUseStatements(ClassMetadata $class): string
  333.     {
  334.         if (PHP_VERSION_ID 80100) {
  335.             return "\n";
  336.         }
  337.         $defaultProperties          $class->getReflectionClass()->getDefaultProperties();
  338.         $lazyLoadedPublicProperties $this->getLazyLoadedPublicPropertiesNames($class);
  339.         $enumClasses                = [];
  340.         foreach ($class->getReflectionClass()->getProperties(ReflectionProperty::IS_PUBLIC) as $property) {
  341.             $name $property->getName();
  342.             if (! in_array($name$lazyLoadedPublicPropertiestrue)) {
  343.                 continue;
  344.             }
  345.             if (array_key_exists($name$defaultProperties) && $defaultProperties[$name] instanceof BackedEnum) {
  346.                 $enumClassNameParts explode('\\'get_class($defaultProperties[$name]));
  347.                 $enumClasses[]      = $enumClassNameParts[0];
  348.             }
  349.         }
  350.         return implode(
  351.             "\n",
  352.             array_map(
  353.                 static function ($className) {
  354.                     return 'use ' $className ';';
  355.                 },
  356.                 array_unique($enumClasses)
  357.             )
  358.         ) . "\n";
  359.     }
  360.     /**
  361.      * Generates the original class name.
  362.      *
  363.      * @return string
  364.      */
  365.     private function generateClassName(ClassMetadata $class)
  366.     {
  367.         return ltrim($class->getName(), '\\');
  368.     }
  369.     /**
  370.      * Generates the array representation of lazy loaded public properties and their default values.
  371.      *
  372.      * @return string
  373.      */
  374.     private function generateLazyPropertiesNames(ClassMetadata $class)
  375.     {
  376.         $lazyPublicProperties $this->getLazyLoadedPublicPropertiesNames($class);
  377.         $values               = [];
  378.         foreach ($lazyPublicProperties as $name) {
  379.             $values[$name] = null;
  380.         }
  381.         return var_export($valuestrue);
  382.     }
  383.     /**
  384.      * Generates the array representation of lazy loaded public properties names.
  385.      *
  386.      * @return string
  387.      */
  388.     private function generateLazyPropertiesDefaults(ClassMetadata $class)
  389.     {
  390.         return var_export($this->getLazyLoadedPublicProperties($class), true);
  391.     }
  392.     /**
  393.      * Generates the constructor code (un-setting public lazy loaded properties, setting identifier field values).
  394.      *
  395.      * @return string
  396.      */
  397.     private function generateConstructorImpl(ClassMetadata $class)
  398.     {
  399.         $constructorImpl = <<<'EOT'
  400.     public function __construct(?\Closure $initializer = null, ?\Closure $cloner = null)
  401.     {
  402. EOT;
  403.         $toUnset array_map(static function (string $name): string {
  404.             return '$this->' $name;
  405.         }, $this->getLazyLoadedPublicPropertiesNames($class));
  406.         return $constructorImpl . ($toUnset === [] ? '' '        unset(' implode(', '$toUnset) . ");\n")
  407.             . <<<'EOT'
  408.         $this->__initializer__ = $initializer;
  409.         $this->__cloner__      = $cloner;
  410.     }
  411. EOT;
  412.     }
  413.     /**
  414.      * Generates the magic getter invoked when lazy loaded public properties are requested.
  415.      *
  416.      * @return string
  417.      */
  418.     private function generateMagicGet(ClassMetadata $class)
  419.     {
  420.         $lazyPublicProperties $this->getLazyLoadedPublicPropertiesNames($class);
  421.         $reflectionClass      $class->getReflectionClass();
  422.         $hasParentGet         false;
  423.         $returnReference      '';
  424.         $inheritDoc           '';
  425.         $name                 '$name';
  426.         $parametersString     '$name';
  427.         $returnTypeHint       null;
  428.         if ($reflectionClass->hasMethod('__get')) {
  429.             $hasParentGet     true;
  430.             $inheritDoc       '{@inheritDoc}';
  431.             $methodReflection $reflectionClass->getMethod('__get');
  432.             if ($methodReflection->returnsReference()) {
  433.                 $returnReference '& ';
  434.             }
  435.             $methodParameters $methodReflection->getParameters();
  436.             $name             '$' $methodParameters[0]->getName();
  437.             $parametersString $this->buildParametersString($methodReflection->getParameters(), ['name']);
  438.             $returnTypeHint   $this->getMethodReturnType($methodReflection);
  439.         }
  440.         if (empty($lazyPublicProperties) && ! $hasParentGet) {
  441.             return '';
  442.         }
  443.         $magicGet = <<<EOT
  444.     /**
  445.      * $inheritDoc
  446.      * @param string \$name
  447.      */
  448.     public function {$returnReference}__get($parametersString)$returnTypeHint
  449.     {
  450. EOT;
  451.         if (! empty($lazyPublicProperties)) {
  452.             $magicGet .= <<<'EOT'
  453.         if (\array_key_exists($name, self::$lazyPropertiesNames)) {
  454.             $this->__initializer__ && $this->__initializer__->__invoke($this, '__get', [$name]);
  455. EOT;
  456.             if ($returnTypeHint === ': void') {
  457.                 $magicGet .= "\n            return;";
  458.             } else {
  459.                 $magicGet .= "\n            return \$this->\$name;";
  460.             }
  461.             $magicGet .= <<<'EOT'
  462.         }
  463. EOT;
  464.         }
  465.         if ($hasParentGet) {
  466.             $magicGet .= <<<'EOT'
  467.         $this->__initializer__ && $this->__initializer__->__invoke($this, '__get', [$name]);
  468. EOT;
  469.             if ($returnTypeHint === ': void') {
  470.                 $magicGet .= <<<'EOT'
  471.         parent::__get($name);
  472.         return;
  473. EOT;
  474.             } elseif ($returnTypeHint === ': never') {
  475.                 $magicGet .= <<<'EOT'
  476.         parent::__get($name);
  477. EOT;
  478.             } else {
  479.                 $magicGet .= <<<'EOT'
  480.         return parent::__get($name);
  481. EOT;
  482.             }
  483.         } else {
  484.             $magicGet .= sprintf(<<<EOT
  485.         trigger_error(sprintf('Undefined property: %%s::$%%s', __CLASS__, %s), E_USER_NOTICE);
  486. EOT
  487.                 , $name);
  488.         }
  489.         return $magicGet "\n    }";
  490.     }
  491.     /**
  492.      * Generates the magic setter (currently unused).
  493.      *
  494.      * @return string
  495.      */
  496.     private function generateMagicSet(ClassMetadata $class)
  497.     {
  498.         $lazyPublicProperties $this->getLazyLoadedPublicPropertiesNames($class);
  499.         $reflectionClass      $class->getReflectionClass();
  500.         $hasParentSet         false;
  501.         $inheritDoc           '';
  502.         $parametersString     '$name, $value';
  503.         $returnTypeHint       null;
  504.         if ($reflectionClass->hasMethod('__set')) {
  505.             $hasParentSet     true;
  506.             $inheritDoc       '{@inheritDoc}';
  507.             $methodReflection $reflectionClass->getMethod('__set');
  508.             $parametersString $this->buildParametersString($methodReflection->getParameters(), ['name''value']);
  509.             $returnTypeHint   $this->getMethodReturnType($methodReflection);
  510.         }
  511.         if (empty($lazyPublicProperties) && ! $hasParentSet) {
  512.             return '';
  513.         }
  514.         $magicSet = <<<EOT
  515.     /**
  516.      * $inheritDoc
  517.      * @param string \$name
  518.      * @param mixed  \$value
  519.      */
  520.     public function __set($parametersString)$returnTypeHint
  521.     {
  522. EOT;
  523.         if (! empty($lazyPublicProperties)) {
  524.             $magicSet .= <<<'EOT'
  525.         if (\array_key_exists($name, self::$lazyPropertiesNames)) {
  526.             $this->__initializer__ && $this->__initializer__->__invoke($this, '__set', [$name, $value]);
  527.             $this->$name = $value;
  528.             return;
  529.         }
  530. EOT;
  531.         }
  532.         if ($hasParentSet) {
  533.             $magicSet .= <<<'EOT'
  534.         $this->__initializer__ && $this->__initializer__->__invoke($this, '__set', [$name, $value]);
  535. EOT;
  536.             if ($returnTypeHint === ': void') {
  537.                 $magicSet .= <<<'EOT'
  538.         parent::__set($name, $value);
  539.         return;
  540. EOT;
  541.             } elseif ($returnTypeHint === ': never') {
  542.                 $magicSet .= <<<'EOT'
  543.         parent::__set($name, $value);
  544. EOT;
  545.             } else {
  546.                 $magicSet .= <<<'EOT'
  547.         return parent::__set($name, $value);
  548. EOT;
  549.             }
  550.         } else {
  551.             $magicSet .= '        $this->$name = $value;';
  552.         }
  553.         return $magicSet "\n    }";
  554.     }
  555.     /**
  556.      * Generates the magic issetter invoked when lazy loaded public properties are checked against isset().
  557.      *
  558.      * @return string
  559.      */
  560.     private function generateMagicIsset(ClassMetadata $class)
  561.     {
  562.         $lazyPublicProperties $this->getLazyLoadedPublicPropertiesNames($class);
  563.         $hasParentIsset       $class->getReflectionClass()->hasMethod('__isset');
  564.         $parametersString     '$name';
  565.         $returnTypeHint       null;
  566.         if ($hasParentIsset) {
  567.             $methodReflection $class->getReflectionClass()->getMethod('__isset');
  568.             $parametersString $this->buildParametersString($methodReflection->getParameters(), ['name']);
  569.             $returnTypeHint   $this->getMethodReturnType($methodReflection);
  570.         }
  571.         if (empty($lazyPublicProperties) && ! $hasParentIsset) {
  572.             return '';
  573.         }
  574.         $inheritDoc $hasParentIsset '{@inheritDoc}' '';
  575.         $magicIsset = <<<EOT
  576.     /**
  577.      * $inheritDoc
  578.      * @param  string \$name
  579.      * @return boolean
  580.      */
  581.     public function __isset($parametersString)$returnTypeHint
  582.     {
  583. EOT;
  584.         if (! empty($lazyPublicProperties)) {
  585.             $magicIsset .= <<<'EOT'
  586.         if (\array_key_exists($name, self::$lazyPropertiesNames)) {
  587.             $this->__initializer__ && $this->__initializer__->__invoke($this, '__isset', [$name]);
  588.             return isset($this->$name);
  589.         }
  590. EOT;
  591.         }
  592.         if ($hasParentIsset) {
  593.             $magicIsset .= <<<'EOT'
  594.         $this->__initializer__ && $this->__initializer__->__invoke($this, '__isset', [$name]);
  595.         return parent::__isset($name);
  596. EOT;
  597.         } else {
  598.             $magicIsset .= '        return false;';
  599.         }
  600.         return $magicIsset "\n    }";
  601.     }
  602.     /**
  603.      * Generates implementation for the `__sleep` method of proxies.
  604.      *
  605.      * @return string
  606.      */
  607.     private function generateSleepImpl(ClassMetadata $class)
  608.     {
  609.         $reflectionClass $class->getReflectionClass();
  610.         $hasParentSleep $reflectionClass->hasMethod('__sleep');
  611.         $inheritDoc     $hasParentSleep '{@inheritDoc}' '';
  612.         $returnTypeHint $hasParentSleep $this->getMethodReturnType($reflectionClass->getMethod('__sleep')) : '';
  613.         $sleepImpl      = <<<EOT
  614.     /**
  615.      * $inheritDoc
  616.      * @return array
  617.      */
  618.     public function __sleep()$returnTypeHint
  619.     {
  620. EOT;
  621.         if ($hasParentSleep) {
  622.             return $sleepImpl . <<<'EOT'
  623.         $properties = array_merge(['__isInitialized__'], parent::__sleep());
  624.         if ($this->__isInitialized__) {
  625.             $properties = array_diff($properties, array_keys(self::$lazyPropertiesNames));
  626.         }
  627.         return $properties;
  628.     }
  629. EOT;
  630.         }
  631.         $allProperties = ['__isInitialized__'];
  632.         foreach ($class->getReflectionClass()->getProperties() as $prop) {
  633.             assert($prop instanceof ReflectionProperty);
  634.             if ($prop->isStatic()) {
  635.                 continue;
  636.             }
  637.             $allProperties[] = $prop->isPrivate()
  638.                 ? "\0" $prop->getDeclaringClass()->getName() . "\0" $prop->getName()
  639.                 : $prop->getName();
  640.         }
  641.         $lazyPublicProperties $this->getLazyLoadedPublicPropertiesNames($class);
  642.         $protectedProperties  array_diff($allProperties$lazyPublicProperties);
  643.         foreach ($allProperties as &$property) {
  644.             $property var_export($propertytrue);
  645.         }
  646.         foreach ($protectedProperties as &$property) {
  647.             $property var_export($propertytrue);
  648.         }
  649.         $allProperties       implode(', '$allProperties);
  650.         $protectedProperties implode(', '$protectedProperties);
  651.         return $sleepImpl . <<<EOT
  652.         if (\$this->__isInitialized__) {
  653.             return [$allProperties];
  654.         }
  655.         return [$protectedProperties];
  656.     }
  657. EOT;
  658.     }
  659.     /**
  660.      * Generates implementation for the `__wakeup` method of proxies.
  661.      *
  662.      * @return string
  663.      */
  664.     private function generateWakeupImpl(ClassMetadata $class)
  665.     {
  666.         $reflectionClass $class->getReflectionClass();
  667.         $hasParentWakeup $reflectionClass->hasMethod('__wakeup');
  668.         $unsetPublicProperties = [];
  669.         foreach ($this->getLazyLoadedPublicPropertiesNames($class) as $lazyPublicProperty) {
  670.             $unsetPublicProperties[] = '$this->' $lazyPublicProperty;
  671.         }
  672.         $shortName      $this->generateProxyShortClassName($class);
  673.         $inheritDoc     $hasParentWakeup '{@inheritDoc}' '';
  674.         $returnTypeHint $hasParentWakeup $this->getMethodReturnType($reflectionClass->getMethod('__wakeup')) : '';
  675.         $wakeupImpl     = <<<EOT
  676.     /**
  677.      * $inheritDoc
  678.      */
  679.     public function __wakeup()$returnTypeHint
  680.     {
  681.         if ( ! \$this->__isInitialized__) {
  682.             \$this->__initializer__ = function ($shortName \$proxy) {
  683.                 \$proxy->__setInitializer(null);
  684.                 \$proxy->__setCloner(null);
  685.                 \$existingProperties = get_object_vars(\$proxy);
  686.                 foreach (\$proxy::\$lazyPropertiesDefaults as \$property => \$defaultValue) {
  687.                     if ( ! array_key_exists(\$property, \$existingProperties)) {
  688.                         \$proxy->\$property = \$defaultValue;
  689.                     }
  690.                 }
  691.             };
  692. EOT;
  693.         if (! empty($unsetPublicProperties)) {
  694.             $wakeupImpl .= "\n            unset(" implode(', '$unsetPublicProperties) . ');';
  695.         }
  696.         $wakeupImpl .= "\n        }";
  697.         if ($hasParentWakeup) {
  698.             $wakeupImpl .= "\n        parent::__wakeup();";
  699.         }
  700.         $wakeupImpl .= "\n    }";
  701.         return $wakeupImpl;
  702.     }
  703.     /**
  704.      * Generates implementation for the `__clone` method of proxies.
  705.      *
  706.      * @return string
  707.      */
  708.     private function generateCloneImpl(ClassMetadata $class)
  709.     {
  710.         $hasParentClone  $class->getReflectionClass()->hasMethod('__clone');
  711.         $inheritDoc      $hasParentClone '{@inheritDoc}' '';
  712.         $callParentClone $hasParentClone "\n        parent::__clone();\n" '';
  713.         return <<<EOT
  714.     /**
  715.      * $inheritDoc
  716.      */
  717.     public function __clone()
  718.     {
  719.         \$this->__cloner__ && \$this->__cloner__->__invoke(\$this, '__clone', []);
  720. $callParentClone    }
  721. EOT;
  722.     }
  723.     /**
  724.      * Generates decorated methods by picking those available in the parent class.
  725.      *
  726.      * @return string
  727.      */
  728.     private function generateMethods(ClassMetadata $class)
  729.     {
  730.         $methods           '';
  731.         $methodNames       = [];
  732.         $reflectionMethods $class->getReflectionClass()->getMethods(ReflectionMethod::IS_PUBLIC);
  733.         $skippedMethods    = [
  734.             '__sleep'   => true,
  735.             '__clone'   => true,
  736.             '__wakeup'  => true,
  737.             '__get'     => true,
  738.             '__set'     => true,
  739.             '__isset'   => true,
  740.         ];
  741.         foreach ($reflectionMethods as $method) {
  742.             $name $method->getName();
  743.             if (
  744.                 $method->isConstructor() ||
  745.                 isset($skippedMethods[strtolower($name)]) ||
  746.                 isset($methodNames[$name]) ||
  747.                 $method->isFinal() ||
  748.                 $method->isStatic() ||
  749.                 ( ! $method->isPublic())
  750.             ) {
  751.                 continue;
  752.             }
  753.             $methodNames[$name] = true;
  754.             $methods           .= "\n    /**\n"
  755.                 "     * {@inheritDoc}\n"
  756.                 "     */\n"
  757.                 '    public function ';
  758.             if ($method->returnsReference()) {
  759.                 $methods .= '&';
  760.             }
  761.             $methods .= $name '(' $this->buildParametersString($method->getParameters()) . ')';
  762.             $methods .= $this->getMethodReturnType($method);
  763.             $methods .= "\n" '    {' "\n";
  764.             if ($this->isShortIdentifierGetter($method$class)) {
  765.                 $identifier lcfirst(substr($name3));
  766.                 $fieldType  $class->getTypeOfField($identifier);
  767.                 $cast       in_array($fieldType, ['integer''smallint']) ? '(int) ' '';
  768.                 $methods .= '        if ($this->__isInitialized__ === false) {' "\n";
  769.                 $methods .= '            ';
  770.                 $methods .= $this->shouldProxiedMethodReturn($method) ? 'return ' '';
  771.                 $methods .= $cast ' parent::' $method->getName() . "();\n";
  772.                 $methods .= '        }' "\n\n";
  773.             }
  774.             $invokeParamsString implode(', '$this->getParameterNamesForInvoke($method->getParameters()));
  775.             $callParamsString   implode(', '$this->getParameterNamesForParentCall($method->getParameters()));
  776.             $methods .= "\n        \$this->__initializer__ "
  777.                 '&& $this->__initializer__->__invoke($this, ' var_export($nametrue)
  778.                 . ', [' $invokeParamsString ']);'
  779.                 "\n\n        "
  780.                 . ($this->shouldProxiedMethodReturn($method) ? 'return ' '')
  781.                 . 'parent::' $name '(' $callParamsString ');'
  782.                 "\n" '    }' "\n";
  783.         }
  784.         return $methods;
  785.     }
  786.     /**
  787.      * Generates the Proxy file name.
  788.      *
  789.      * @param string $className
  790.      * @param string $baseDirectory Optional base directory for proxy file name generation.
  791.      *                              If not specified, the directory configured on the Configuration of the
  792.      *                              EntityManager will be used by this factory.
  793.      * @psalm-param class-string $className
  794.      *
  795.      * @return string
  796.      */
  797.     public function getProxyFileName($className$baseDirectory null)
  798.     {
  799.         $baseDirectory $baseDirectory ?: $this->proxyDirectory;
  800.         return rtrim($baseDirectoryDIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR Proxy::MARKER
  801.             str_replace('\\'''$className) . '.php';
  802.     }
  803.     /**
  804.      * Checks if the method is a short identifier getter.
  805.      *
  806.      * What does this mean? For proxy objects the identifier is already known,
  807.      * however accessing the getter for this identifier usually triggers the
  808.      * lazy loading, leading to a query that may not be necessary if only the
  809.      * ID is interesting for the userland code (for example in views that
  810.      * generate links to the entity, but do not display anything else).
  811.      *
  812.      * @param ReflectionMethod $method
  813.      *
  814.      * @return bool
  815.      */
  816.     private function isShortIdentifierGetter($methodClassMetadata $class)
  817.     {
  818.         $identifier lcfirst(substr($method->getName(), 3));
  819.         $startLine  $method->getStartLine();
  820.         $endLine    $method->getEndLine();
  821.         $cheapCheck $method->getNumberOfParameters() === 0
  822.             && substr($method->getName(), 03) === 'get'
  823.             && in_array($identifier$class->getIdentifier(), true)
  824.             && $class->hasField($identifier)
  825.             && ($endLine $startLine <= 4);
  826.         if ($cheapCheck) {
  827.             $code file($method->getFileName());
  828.             $code trim(implode(' 'array_slice($code$startLine 1$endLine $startLine 1)));
  829.             $pattern sprintf(self::PATTERN_MATCH_ID_METHOD$method->getName(), $identifier);
  830.             if (preg_match($pattern$code)) {
  831.                 return true;
  832.             }
  833.         }
  834.         return false;
  835.     }
  836.     /**
  837.      * Generates the list of public properties to be lazy loaded.
  838.      *
  839.      * @return array<int, string>
  840.      */
  841.     private function getLazyLoadedPublicPropertiesNames(ClassMetadata $class): array
  842.     {
  843.         $properties = [];
  844.         foreach ($class->getReflectionClass()->getProperties(ReflectionProperty::IS_PUBLIC) as $property) {
  845.             $name $property->getName();
  846.             if ((! $class->hasField($name) && ! $class->hasAssociation($name)) || $class->isIdentifier($name)) {
  847.                 continue;
  848.             }
  849.             $properties[] = $name;
  850.         }
  851.         return $properties;
  852.     }
  853.     /**
  854.      * Generates the list of default values of public properties.
  855.      *
  856.      * @return mixed[]
  857.      */
  858.     private function getLazyLoadedPublicProperties(ClassMetadata $class)
  859.     {
  860.         $defaultProperties          $class->getReflectionClass()->getDefaultProperties();
  861.         $lazyLoadedPublicProperties $this->getLazyLoadedPublicPropertiesNames($class);
  862.         $defaultValues              = [];
  863.         foreach ($class->getReflectionClass()->getProperties(ReflectionProperty::IS_PUBLIC) as $property) {
  864.             $name $property->getName();
  865.             if (! in_array($name$lazyLoadedPublicPropertiestrue)) {
  866.                 continue;
  867.             }
  868.             if (array_key_exists($name$defaultProperties)) {
  869.                 $defaultValues[$name] = $defaultProperties[$name];
  870.             } elseif (method_exists($property'getType')) {
  871.                 $propertyType $property->getType();
  872.                 if ($propertyType !== null && $propertyType->allowsNull()) {
  873.                     $defaultValues[$name] = null;
  874.                 }
  875.             }
  876.         }
  877.         return $defaultValues;
  878.     }
  879.     /**
  880.      * @param ReflectionParameter[] $parameters
  881.      * @param string[]              $renameParameters
  882.      *
  883.      * @return string
  884.      */
  885.     private function buildParametersString(array $parameters, array $renameParameters = [])
  886.     {
  887.         $parameterDefinitions = [];
  888.         $i = -1;
  889.         foreach ($parameters as $param) {
  890.             assert($param instanceof ReflectionParameter);
  891.             $i++;
  892.             $parameterDefinition '';
  893.             $parameterType       $this->getParameterType($param);
  894.             if ($parameterType !== null) {
  895.                 $parameterDefinition .= $parameterType ' ';
  896.             }
  897.             if ($param->isPassedByReference()) {
  898.                 $parameterDefinition .= '&';
  899.             }
  900.             if ($param->isVariadic()) {
  901.                 $parameterDefinition .= '...';
  902.             }
  903.             $parameterDefinition .= '$' . ($renameParameters $renameParameters[$i] : $param->getName());
  904.             if ($param->isDefaultValueAvailable()) {
  905.                 $parameterDefinition .= ' = ' var_export($param->getDefaultValue(), true);
  906.             }
  907.             $parameterDefinitions[] = $parameterDefinition;
  908.         }
  909.         return implode(', '$parameterDefinitions);
  910.     }
  911.     /**
  912.      * @return string|null
  913.      */
  914.     private function getParameterType(ReflectionParameter $parameter)
  915.     {
  916.         if (! $parameter->hasType()) {
  917.             return null;
  918.         }
  919.         $declaringFunction $parameter->getDeclaringFunction();
  920.         assert($declaringFunction instanceof ReflectionMethod);
  921.         return $this->formatType($parameter->getType(), $declaringFunction$parameter);
  922.     }
  923.     /**
  924.      * @param ReflectionParameter[] $parameters
  925.      *
  926.      * @return string[]
  927.      */
  928.     private function getParameterNamesForInvoke(array $parameters)
  929.     {
  930.         return array_map(
  931.             static function (ReflectionParameter $parameter) {
  932.                 return '$' $parameter->getName();
  933.             },
  934.             $parameters
  935.         );
  936.     }
  937.     /**
  938.      * @param ReflectionParameter[] $parameters
  939.      *
  940.      * @return string[]
  941.      */
  942.     private function getParameterNamesForParentCall(array $parameters)
  943.     {
  944.         return array_map(
  945.             static function (ReflectionParameter $parameter) {
  946.                 $name '';
  947.                 if ($parameter->isVariadic()) {
  948.                     $name .= '...';
  949.                 }
  950.                 $name .= '$' $parameter->getName();
  951.                 return $name;
  952.             },
  953.             $parameters
  954.         );
  955.     }
  956.     /**
  957.      * @return string
  958.      */
  959.     private function getMethodReturnType(ReflectionMethod $method)
  960.     {
  961.         if (! $method->hasReturnType()) {
  962.             return '';
  963.         }
  964.         return ': ' $this->formatType($method->getReturnType(), $method);
  965.     }
  966.     /**
  967.      * @return bool
  968.      */
  969.     private function shouldProxiedMethodReturn(ReflectionMethod $method)
  970.     {
  971.         if (! $method->hasReturnType()) {
  972.             return true;
  973.         }
  974.         return ! in_array(
  975.             strtolower($this->formatType($method->getReturnType(), $method)),
  976.             ['void''never'],
  977.             true
  978.         );
  979.     }
  980.     /**
  981.      * @return string
  982.      */
  983.     private function formatType(
  984.         ReflectionType $type,
  985.         ReflectionMethod $method,
  986.         ?ReflectionParameter $parameter null
  987.     ) {
  988.         if ($type instanceof ReflectionUnionType) {
  989.             return implode('|'array_map(
  990.                 function (ReflectionType $unionedType) use ($method$parameter) {
  991.                     return $this->formatType($unionedType$method$parameter);
  992.                 },
  993.                 $type->getTypes()
  994.             ));
  995.         }
  996.         if ($type instanceof ReflectionIntersectionType) {
  997.             return implode('&'array_map(
  998.                 function (ReflectionType $intersectedType) use ($method$parameter) {
  999.                     return $this->formatType($intersectedType$method$parameter);
  1000.                 },
  1001.                 $type->getTypes()
  1002.             ));
  1003.         }
  1004.         assert($type instanceof ReflectionNamedType);
  1005.         $name      $type->getName();
  1006.         $nameLower strtolower($name);
  1007.         if ($nameLower === 'static') {
  1008.             $name 'static';
  1009.         }
  1010.         if ($nameLower === 'self') {
  1011.             $name $method->getDeclaringClass()->getName();
  1012.         }
  1013.         if ($nameLower === 'parent') {
  1014.             $name $method->getDeclaringClass()->getParentClass()->getName();
  1015.         }
  1016.         if (! $type->isBuiltin() && ! class_exists($name) && ! interface_exists($name) && $name !== 'static') {
  1017.             if ($parameter !== null) {
  1018.                 throw UnexpectedValueException::invalidParameterTypeHint(
  1019.                     $method->getDeclaringClass()->getName(),
  1020.                     $method->getName(),
  1021.                     $parameter->getName()
  1022.                 );
  1023.             }
  1024.             throw UnexpectedValueException::invalidReturnTypeHint(
  1025.                 $method->getDeclaringClass()->getName(),
  1026.                 $method->getName()
  1027.             );
  1028.         }
  1029.         if (! $type->isBuiltin() && $name !== 'static') {
  1030.             $name '\\' $name;
  1031.         }
  1032.         if (
  1033.             $type->allowsNull()
  1034.             && ! in_array($name, ['mixed''null'], true)
  1035.             && ($parameter === null || ! $parameter->isDefaultValueAvailable() || $parameter->getDefaultValue() !== null)
  1036.         ) {
  1037.             $name '?' $name;
  1038.         }
  1039.         return $name;
  1040.     }
  1041. }