123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928 |
- <?php
- /*
- * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
- * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
- * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
- * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
- * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
- * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
- * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
- * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
- * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
- * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
- * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
- *
- * This software consists of voluntary contributions made by many individuals
- * and is licensed under the MIT license. For more information, see
- * <http://www.doctrine-project.org>.
- */
- namespace Doctrine\Common\Proxy;
- use Doctrine\Common\Persistence\Mapping\ClassMetadata;
- use Doctrine\Common\Util\ClassUtils;
- use Doctrine\Common\Proxy\Exception\InvalidArgumentException;
- use Doctrine\Common\Proxy\Exception\UnexpectedValueException;
- /**
- * This factory is used to generate proxy classes.
- * It builds proxies from given parameters, a template and class metadata.
- *
- * @author Marco Pivetta <ocramius@gmail.com>
- * @since 2.4
- */
- class ProxyGenerator
- {
- /**
- * Used to match very simple id methods that don't need
- * to be decorated since the identifier is known.
- */
- const PATTERN_MATCH_ID_METHOD = '((public\s)?(function\s{1,}%s\s?\(\)\s{1,})\s{0,}{\s{0,}return\s{0,}\$this->%s;\s{0,}})i';
- /**
- * The namespace that contains all proxy classes.
- *
- * @var string
- */
- private $proxyNamespace;
- /**
- * The directory that contains all proxy classes.
- *
- * @var string
- */
- private $proxyDirectory;
- /**
- * Map of callables used to fill in placeholders set in the template.
- *
- * @var string[]|callable[]
- */
- protected $placeholders = array(
- 'baseProxyInterface' => 'Doctrine\Common\Proxy\Proxy',
- 'additionalProperties' => '',
- );
- /**
- * Template used as a blueprint to generate proxies.
- *
- * @var string
- */
- protected $proxyClassTemplate = '<?php
- namespace <namespace>;
- /**
- * DO NOT EDIT THIS FILE - IT WAS CREATED BY DOCTRINE\'S PROXY GENERATOR
- */
- class <proxyShortClassName> extends \<className> implements \<baseProxyInterface>
- {
- /**
- * @var \Closure the callback responsible for loading properties in the proxy object. This callback is called with
- * three parameters, being respectively the proxy object to be initialized, the method that triggered the
- * initialization process and an array of ordered parameters that were passed to that method.
- *
- * @see \Doctrine\Common\Persistence\Proxy::__setInitializer
- */
- public $__initializer__;
- /**
- * @var \Closure the callback responsible of loading properties that need to be copied in the cloned object
- *
- * @see \Doctrine\Common\Persistence\Proxy::__setCloner
- */
- public $__cloner__;
- /**
- * @var boolean flag indicating if this object was already initialized
- *
- * @see \Doctrine\Common\Persistence\Proxy::__isInitialized
- */
- public $__isInitialized__ = false;
- /**
- * @var array properties to be lazy loaded, with keys being the property
- * names and values being their default values
- *
- * @see \Doctrine\Common\Persistence\Proxy::__getLazyProperties
- */
- public static $lazyPropertiesDefaults = array(<lazyPropertiesDefaults>);
- <additionalProperties>
- <constructorImpl>
- <magicGet>
- <magicSet>
- <magicIsset>
- <sleepImpl>
- <wakeupImpl>
- <cloneImpl>
- /**
- * Forces initialization of the proxy
- */
- public function __load()
- {
- $this->__initializer__ && $this->__initializer__->__invoke($this, \'__load\', array());
- }
- /**
- * {@inheritDoc}
- * @internal generated method: use only when explicitly handling proxy specific loading logic
- */
- public function __isInitialized()
- {
- return $this->__isInitialized__;
- }
- /**
- * {@inheritDoc}
- * @internal generated method: use only when explicitly handling proxy specific loading logic
- */
- public function __setInitialized($initialized)
- {
- $this->__isInitialized__ = $initialized;
- }
- /**
- * {@inheritDoc}
- * @internal generated method: use only when explicitly handling proxy specific loading logic
- */
- public function __setInitializer(\Closure $initializer = null)
- {
- $this->__initializer__ = $initializer;
- }
- /**
- * {@inheritDoc}
- * @internal generated method: use only when explicitly handling proxy specific loading logic
- */
- public function __getInitializer()
- {
- return $this->__initializer__;
- }
- /**
- * {@inheritDoc}
- * @internal generated method: use only when explicitly handling proxy specific loading logic
- */
- public function __setCloner(\Closure $cloner = null)
- {
- $this->__cloner__ = $cloner;
- }
- /**
- * {@inheritDoc}
- * @internal generated method: use only when explicitly handling proxy specific cloning logic
- */
- public function __getCloner()
- {
- return $this->__cloner__;
- }
- /**
- * {@inheritDoc}
- * @internal generated method: use only when explicitly handling proxy specific loading logic
- * @static
- */
- public function __getLazyProperties()
- {
- return self::$lazyPropertiesDefaults;
- }
- <methods>
- }
- ';
- /**
- * Initializes a new instance of the <tt>ProxyFactory</tt> class that is
- * connected to the given <tt>EntityManager</tt>.
- *
- * @param string $proxyDirectory The directory to use for the proxy classes. It must exist.
- * @param string $proxyNamespace The namespace to use for the proxy classes.
- *
- * @throws InvalidArgumentException
- */
- public function __construct($proxyDirectory, $proxyNamespace)
- {
- if ( ! $proxyDirectory) {
- throw InvalidArgumentException::proxyDirectoryRequired();
- }
- if ( ! $proxyNamespace) {
- throw InvalidArgumentException::proxyNamespaceRequired();
- }
- $this->proxyDirectory = $proxyDirectory;
- $this->proxyNamespace = $proxyNamespace;
- }
- /**
- * Sets a placeholder to be replaced in the template.
- *
- * @param string $name
- * @param string|callable $placeholder
- *
- * @throws InvalidArgumentException
- */
- public function setPlaceholder($name, $placeholder)
- {
- if ( ! is_string($placeholder) && ! is_callable($placeholder)) {
- throw InvalidArgumentException::invalidPlaceholder($name);
- }
- $this->placeholders[$name] = $placeholder;
- }
- /**
- * Sets the base template used to create proxy classes.
- *
- * @param string $proxyClassTemplate
- */
- public function setProxyClassTemplate($proxyClassTemplate)
- {
- $this->proxyClassTemplate = (string) $proxyClassTemplate;
- }
- /**
- * Generates a proxy class file.
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class Metadata for the original class.
- * @param string|bool $fileName Filename (full path) for the generated class. If none is given, eval() is used.
- *
- * @throws UnexpectedValueException
- */
- public function generateProxyClass(ClassMetadata $class, $fileName = false)
- {
- preg_match_all('(<([a-zA-Z]+)>)', $this->proxyClassTemplate, $placeholderMatches);
- $placeholderMatches = array_combine($placeholderMatches[0], $placeholderMatches[1]);
- $placeholders = array();
- foreach ($placeholderMatches as $placeholder => $name) {
- $placeholders[$placeholder] = isset($this->placeholders[$name])
- ? $this->placeholders[$name]
- : array($this, 'generate' . $name);
- }
- foreach ($placeholders as & $placeholder) {
- if (is_callable($placeholder)) {
- $placeholder = call_user_func($placeholder, $class);
- }
- }
- $proxyCode = strtr($this->proxyClassTemplate, $placeholders);
- if ( ! $fileName) {
- $proxyClassName = $this->generateNamespace($class) . '\\' . $this->generateProxyShortClassName($class);
- if ( ! class_exists($proxyClassName)) {
- eval(substr($proxyCode, 5));
- }
- return;
- }
- $parentDirectory = dirname($fileName);
- if ( ! is_dir($parentDirectory) && (false === @mkdir($parentDirectory, 0775, true))) {
- throw UnexpectedValueException::proxyDirectoryNotWritable();
- }
- if ( ! is_writable($parentDirectory)) {
- throw UnexpectedValueException::proxyDirectoryNotWritable();
- }
- $tmpFileName = $fileName . '.' . uniqid('', true);
- file_put_contents($tmpFileName, $proxyCode);
- rename($tmpFileName, $fileName);
- }
- /**
- * Generates the proxy short class name to be used in the template.
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return string
- */
- private function generateProxyShortClassName(ClassMetadata $class)
- {
- $proxyClassName = ClassUtils::generateProxyClassName($class->getName(), $this->proxyNamespace);
- $parts = explode('\\', strrev($proxyClassName), 2);
- return strrev($parts[0]);
- }
- /**
- * Generates the proxy namespace.
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return string
- */
- private function generateNamespace(ClassMetadata $class)
- {
- $proxyClassName = ClassUtils::generateProxyClassName($class->getName(), $this->proxyNamespace);
- $parts = explode('\\', strrev($proxyClassName), 2);
- return strrev($parts[1]);
- }
- /**
- * Generates the original class name.
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return string
- */
- private function generateClassName(ClassMetadata $class)
- {
- return ltrim($class->getName(), '\\');
- }
- /**
- * Generates the array representation of lazy loaded public properties and their default values.
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return string
- */
- private function generateLazyPropertiesDefaults(ClassMetadata $class)
- {
- $lazyPublicProperties = $this->getLazyLoadedPublicProperties($class);
- $values = array();
- foreach ($lazyPublicProperties as $key => $value) {
- $values[] = var_export($key, true) . ' => ' . var_export($value, true);
- }
- return implode(', ', $values);
- }
- /**
- * Generates the constructor code (un-setting public lazy loaded properties, setting identifier field values).
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return string
- */
- private function generateConstructorImpl(ClassMetadata $class)
- {
- $constructorImpl = <<<'EOT'
- /**
- * @param \Closure $initializer
- * @param \Closure $cloner
- */
- public function __construct($initializer = null, $cloner = null)
- {
- EOT;
- $toUnset = array();
- foreach ($this->getLazyLoadedPublicProperties($class) as $lazyPublicProperty => $unused) {
- $toUnset[] = '$this->' . $lazyPublicProperty;
- }
- $constructorImpl .= (empty($toUnset) ? '' : ' unset(' . implode(', ', $toUnset) . ");\n")
- . <<<'EOT'
- $this->__initializer__ = $initializer;
- $this->__cloner__ = $cloner;
- }
- EOT;
- return $constructorImpl;
- }
- /**
- * Generates the magic getter invoked when lazy loaded public properties are requested.
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return string
- */
- private function generateMagicGet(ClassMetadata $class)
- {
- $lazyPublicProperties = array_keys($this->getLazyLoadedPublicProperties($class));
- $reflectionClass = $class->getReflectionClass();
- $hasParentGet = false;
- $returnReference = '';
- $inheritDoc = '';
- if ($reflectionClass->hasMethod('__get')) {
- $hasParentGet = true;
- $inheritDoc = '{@inheritDoc}';
- if ($reflectionClass->getMethod('__get')->returnsReference()) {
- $returnReference = '& ';
- }
- }
- if (empty($lazyPublicProperties) && ! $hasParentGet) {
- return '';
- }
- $magicGet = <<<EOT
- /**
- * $inheritDoc
- * @param string \$name
- */
- public function {$returnReference}__get(\$name)
- {
- EOT;
- if ( ! empty($lazyPublicProperties)) {
- $magicGet .= <<<'EOT'
- if (array_key_exists($name, $this->__getLazyProperties())) {
- $this->__initializer__ && $this->__initializer__->__invoke($this, '__get', array($name));
- return $this->$name;
- }
- EOT;
- }
- if ($hasParentGet) {
- $magicGet .= <<<'EOT'
- $this->__initializer__ && $this->__initializer__->__invoke($this, '__get', array($name));
- return parent::__get($name);
- EOT;
- } else {
- $magicGet .= <<<'EOT'
- trigger_error(sprintf('Undefined property: %s::$%s', __CLASS__, $name), E_USER_NOTICE);
- EOT;
- }
- $magicGet .= " }";
- return $magicGet;
- }
- /**
- * Generates the magic setter (currently unused).
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return string
- */
- private function generateMagicSet(ClassMetadata $class)
- {
- $lazyPublicProperties = $this->getLazyLoadedPublicProperties($class);
- $hasParentSet = $class->getReflectionClass()->hasMethod('__set');
- if (empty($lazyPublicProperties) && ! $hasParentSet) {
- return '';
- }
- $inheritDoc = $hasParentSet ? '{@inheritDoc}' : '';
- $magicSet = <<<EOT
- /**
- * $inheritDoc
- * @param string \$name
- * @param mixed \$value
- */
- public function __set(\$name, \$value)
- {
- EOT;
- if ( ! empty($lazyPublicProperties)) {
- $magicSet .= <<<'EOT'
- if (array_key_exists($name, $this->__getLazyProperties())) {
- $this->__initializer__ && $this->__initializer__->__invoke($this, '__set', array($name, $value));
- $this->$name = $value;
- return;
- }
- EOT;
- }
- if ($hasParentSet) {
- $magicSet .= <<<'EOT'
- $this->__initializer__ && $this->__initializer__->__invoke($this, '__set', array($name, $value));
- return parent::__set($name, $value);
- EOT;
- } else {
- $magicSet .= " \$this->\$name = \$value;";
- }
- $magicSet .= "\n }";
- return $magicSet;
- }
- /**
- * Generates the magic issetter invoked when lazy loaded public properties are checked against isset().
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return string
- */
- private function generateMagicIsset(ClassMetadata $class)
- {
- $lazyPublicProperties = array_keys($this->getLazyLoadedPublicProperties($class));
- $hasParentIsset = $class->getReflectionClass()->hasMethod('__isset');
- if (empty($lazyPublicProperties) && ! $hasParentIsset) {
- return '';
- }
- $inheritDoc = $hasParentIsset ? '{@inheritDoc}' : '';
- $magicIsset = <<<EOT
- /**
- * $inheritDoc
- * @param string \$name
- * @return boolean
- */
- public function __isset(\$name)
- {
- EOT;
- if ( ! empty($lazyPublicProperties)) {
- $magicIsset .= <<<'EOT'
- if (array_key_exists($name, $this->__getLazyProperties())) {
- $this->__initializer__ && $this->__initializer__->__invoke($this, '__isset', array($name));
- return isset($this->$name);
- }
- EOT;
- }
- if ($hasParentIsset) {
- $magicIsset .= <<<'EOT'
- $this->__initializer__ && $this->__initializer__->__invoke($this, '__isset', array($name));
- return parent::__isset($name);
- EOT;
- } else {
- $magicIsset .= " return false;";
- }
- return $magicIsset . "\n }";
- }
- /**
- * Generates implementation for the `__sleep` method of proxies.
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return string
- */
- private function generateSleepImpl(ClassMetadata $class)
- {
- $hasParentSleep = $class->getReflectionClass()->hasMethod('__sleep');
- $inheritDoc = $hasParentSleep ? '{@inheritDoc}' : '';
- $sleepImpl = <<<EOT
- /**
- * $inheritDoc
- * @return array
- */
- public function __sleep()
- {
- EOT;
- if ($hasParentSleep) {
- return $sleepImpl . <<<'EOT'
- $properties = array_merge(array('__isInitialized__'), parent::__sleep());
- if ($this->__isInitialized__) {
- $properties = array_diff($properties, array_keys($this->__getLazyProperties()));
- }
- return $properties;
- }
- EOT;
- }
- $allProperties = array('__isInitialized__');
- /* @var $prop \ReflectionProperty */
- foreach ($class->getReflectionClass()->getProperties() as $prop) {
- $allProperties[] = $prop->isPrivate()
- ? "\0" . $prop->getDeclaringClass()->getName() . "\0" . $prop->getName()
- : $prop->getName();
- }
- $lazyPublicProperties = array_keys($this->getLazyLoadedPublicProperties($class));
- $protectedProperties = array_diff($allProperties, $lazyPublicProperties);
- foreach ($allProperties as &$property) {
- $property = var_export($property, true);
- }
- foreach ($protectedProperties as &$property) {
- $property = var_export($property, true);
- }
- $allProperties = implode(', ', $allProperties);
- $protectedProperties = implode(', ', $protectedProperties);
- return $sleepImpl . <<<EOT
- if (\$this->__isInitialized__) {
- return array($allProperties);
- }
- return array($protectedProperties);
- }
- EOT;
- }
- /**
- * Generates implementation for the `__wakeup` method of proxies.
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return string
- */
- private function generateWakeupImpl(ClassMetadata $class)
- {
- $unsetPublicProperties = array();
- $hasWakeup = $class->getReflectionClass()->hasMethod('__wakeup');
- foreach (array_keys($this->getLazyLoadedPublicProperties($class)) as $lazyPublicProperty) {
- $unsetPublicProperties[] = '$this->' . $lazyPublicProperty;
- }
- $shortName = $this->generateProxyShortClassName($class);
- $inheritDoc = $hasWakeup ? '{@inheritDoc}' : '';
- $wakeupImpl = <<<EOT
- /**
- * $inheritDoc
- */
- public function __wakeup()
- {
- if ( ! \$this->__isInitialized__) {
- \$this->__initializer__ = function ($shortName \$proxy) {
- \$proxy->__setInitializer(null);
- \$proxy->__setCloner(null);
- \$existingProperties = get_object_vars(\$proxy);
- foreach (\$proxy->__getLazyProperties() as \$property => \$defaultValue) {
- if ( ! array_key_exists(\$property, \$existingProperties)) {
- \$proxy->\$property = \$defaultValue;
- }
- }
- };
- EOT;
- if ( ! empty($unsetPublicProperties)) {
- $wakeupImpl .= "\n unset(" . implode(', ', $unsetPublicProperties) . ");";
- }
- $wakeupImpl .= "\n }";
- if ($hasWakeup) {
- $wakeupImpl .= "\n parent::__wakeup();";
- }
- $wakeupImpl .= "\n }";
- return $wakeupImpl;
- }
- /**
- * Generates implementation for the `__clone` method of proxies.
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return string
- */
- private function generateCloneImpl(ClassMetadata $class)
- {
- $hasParentClone = $class->getReflectionClass()->hasMethod('__clone');
- $inheritDoc = $hasParentClone ? '{@inheritDoc}' : '';
- $callParentClone = $hasParentClone ? "\n parent::__clone();\n" : '';
- return <<<EOT
- /**
- * $inheritDoc
- */
- public function __clone()
- {
- \$this->__cloner__ && \$this->__cloner__->__invoke(\$this, '__clone', array());
- $callParentClone }
- EOT;
- }
- /**
- * Generates decorated methods by picking those available in the parent class.
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return string
- */
- private function generateMethods(ClassMetadata $class)
- {
- $methods = '';
- $methodNames = array();
- $reflectionMethods = $class->getReflectionClass()->getMethods(\ReflectionMethod::IS_PUBLIC);
- $skippedMethods = array(
- '__sleep' => true,
- '__clone' => true,
- '__wakeup' => true,
- '__get' => true,
- '__set' => true,
- '__isset' => true,
- );
- foreach ($reflectionMethods as $method) {
- $name = $method->getName();
- if (
- $method->isConstructor() ||
- isset($skippedMethods[strtolower($name)]) ||
- isset($methodNames[$name]) ||
- $method->isFinal() ||
- $method->isStatic() ||
- ( ! $method->isPublic())
- ) {
- continue;
- }
- $methodNames[$name] = true;
- $methods .= "\n /**\n"
- . " * {@inheritDoc}\n"
- . " */\n"
- . ' public function ';
- if ($method->returnsReference()) {
- $methods .= '&';
- }
- $methods .= $name . '(';
- $firstParam = true;
- $parameterString = '';
- $argumentString = '';
- $parameters = array();
- foreach ($method->getParameters() as $param) {
- if ($firstParam) {
- $firstParam = false;
- } else {
- $parameterString .= ', ';
- $argumentString .= ', ';
- }
- try {
- $paramClass = $param->getClass();
- } catch (\ReflectionException $previous) {
- throw UnexpectedValueException::invalidParameterTypeHint(
- $class->getName(),
- $method->getName(),
- $param->getName(),
- $previous
- );
- }
- // We need to pick the type hint class too
- if (null !== $paramClass) {
- $parameterString .= '\\' . $paramClass->getName() . ' ';
- } elseif ($param->isArray()) {
- $parameterString .= 'array ';
- } elseif (method_exists($param, 'isCallable') && $param->isCallable()) {
- $parameterString .= 'callable ';
- }
- if ($param->isPassedByReference()) {
- $parameterString .= '&';
- }
- $parameters[] = '$' . $param->getName();
- $parameterString .= '$' . $param->getName();
- $argumentString .= '$' . $param->getName();
- if ($param->isDefaultValueAvailable()) {
- $parameterString .= ' = ' . var_export($param->getDefaultValue(), true);
- }
- }
- $methods .= $parameterString . ')';
- $methods .= "\n" . ' {' . "\n";
- if ($this->isShortIdentifierGetter($method, $class)) {
- $identifier = lcfirst(substr($name, 3));
- $fieldType = $class->getTypeOfField($identifier);
- $cast = in_array($fieldType, array('integer', 'smallint')) ? '(int) ' : '';
- $methods .= ' if ($this->__isInitialized__ === false) {' . "\n";
- $methods .= ' return ' . $cast . ' parent::' . $method->getName() . "();\n";
- $methods .= ' }' . "\n\n";
- }
- $methods .= "\n \$this->__initializer__ "
- . "&& \$this->__initializer__->__invoke(\$this, " . var_export($name, true)
- . ", array(" . implode(', ', $parameters) . "));"
- . "\n\n return parent::" . $name . '(' . $argumentString . ');'
- . "\n" . ' }' . "\n";
- }
- return $methods;
- }
- /**
- * Generates the Proxy file name.
- *
- * @param string $className
- * @param string $baseDirectory Optional base directory for proxy file name generation.
- * If not specified, the directory configured on the Configuration of the
- * EntityManager will be used by this factory.
- *
- * @return string
- */
- public function getProxyFileName($className, $baseDirectory = null)
- {
- $baseDirectory = $baseDirectory ?: $this->proxyDirectory;
- return rtrim($baseDirectory, DIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR . Proxy::MARKER
- . str_replace('\\', '', $className) . '.php';
- }
- /**
- * Checks if the method is a short identifier getter.
- *
- * What does this mean? For proxy objects the identifier is already known,
- * however accessing the getter for this identifier usually triggers the
- * lazy loading, leading to a query that may not be necessary if only the
- * ID is interesting for the userland code (for example in views that
- * generate links to the entity, but do not display anything else).
- *
- * @param \ReflectionMethod $method
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return boolean
- */
- private function isShortIdentifierGetter($method, ClassMetadata $class)
- {
- $identifier = lcfirst(substr($method->getName(), 3));
- $startLine = $method->getStartLine();
- $endLine = $method->getEndLine();
- $cheapCheck = (
- $method->getNumberOfParameters() == 0
- && substr($method->getName(), 0, 3) == 'get'
- && in_array($identifier, $class->getIdentifier(), true)
- && $class->hasField($identifier)
- && (($endLine - $startLine) <= 4)
- );
- if ($cheapCheck) {
- $code = file($method->getDeclaringClass()->getFileName());
- $code = trim(implode(' ', array_slice($code, $startLine - 1, $endLine - $startLine + 1)));
- $pattern = sprintf(self::PATTERN_MATCH_ID_METHOD, $method->getName(), $identifier);
- if (preg_match($pattern, $code)) {
- return true;
- }
- }
- return false;
- }
- /**
- * Generates the list of public properties to be lazy loaded, with their default values.
- *
- * @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $class
- *
- * @return mixed[]
- */
- private function getLazyLoadedPublicProperties(ClassMetadata $class)
- {
- $defaultProperties = $class->getReflectionClass()->getDefaultProperties();
- $properties = array();
- foreach ($class->getReflectionClass()->getProperties(\ReflectionProperty::IS_PUBLIC) as $property) {
- $name = $property->getName();
- if (($class->hasField($name) || $class->hasAssociation($name)) && ! $class->isIdentifier($name)) {
- $properties[$name] = $defaultProperties[$name];
- }
- }
- return $properties;
- }
- }
|