2009-07-16 13:29:15 +00:00
|
|
|
<?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
|
2012-05-26 14:37:00 +02:00
|
|
|
* and is licensed under the MIT license. For more information, see
|
2009-07-16 13:29:15 +00:00
|
|
|
* <http://www.doctrine-project.org>.
|
|
|
|
*/
|
|
|
|
|
2009-07-16 14:03:22 +00:00
|
|
|
namespace Doctrine\ORM\Proxy;
|
2009-07-16 13:29:15 +00:00
|
|
|
|
2013-01-06 15:11:30 +01:00
|
|
|
use Doctrine\Common\Persistence\Mapping\ClassMetadata;
|
|
|
|
use Doctrine\Common\Proxy\AbstractProxyFactory;
|
2013-03-01 11:58:51 -06:00
|
|
|
use Doctrine\Common\Proxy\Proxy as BaseProxy;
|
2014-12-05 14:02:47 +01:00
|
|
|
use Doctrine\Common\Proxy\ProxyDefinition;
|
2012-10-20 01:27:53 +02:00
|
|
|
use Doctrine\Common\Proxy\ProxyGenerator;
|
2014-12-05 14:02:47 +01:00
|
|
|
use Doctrine\Common\Util\ClassUtils;
|
|
|
|
use Doctrine\ORM\EntityManagerInterface;
|
2015-01-15 05:01:52 +00:00
|
|
|
use Doctrine\ORM\Persisters\Entity\EntityPersister;
|
2013-01-06 15:11:30 +01:00
|
|
|
use Doctrine\ORM\EntityNotFoundException;
|
2015-01-08 15:59:17 +01:00
|
|
|
use Doctrine\ORM\Utility\IdentifierFlattener;
|
2009-07-16 13:29:15 +00:00
|
|
|
|
|
|
|
/**
|
2009-10-14 20:18:36 +00:00
|
|
|
* This factory is used to create proxy objects for entities at runtime.
|
2009-07-16 13:29:15 +00:00
|
|
|
*
|
|
|
|
* @author Roman Borschel <roman@code-factory.org>
|
2009-07-16 14:03:22 +00:00
|
|
|
* @author Giorgio Sironi <piccoloprincipeazzurro@gmail.com>
|
2013-01-06 15:11:30 +01:00
|
|
|
* @author Marco Pivetta <ocramius@gmail.com>
|
2009-07-16 13:29:15 +00:00
|
|
|
* @since 2.0
|
|
|
|
*/
|
2013-01-06 15:11:30 +01:00
|
|
|
class ProxyFactory extends AbstractProxyFactory
|
2009-07-16 13:29:15 +00:00
|
|
|
{
|
2012-12-13 10:28:55 +00:00
|
|
|
/**
|
2014-12-05 14:02:47 +01:00
|
|
|
* @var EntityManagerInterface The EntityManager this factory is bound to.
|
2012-12-13 10:28:55 +00:00
|
|
|
*/
|
2012-10-20 01:27:53 +02:00
|
|
|
private $em;
|
2012-12-13 10:28:55 +00:00
|
|
|
|
|
|
|
/**
|
2012-10-20 01:27:53 +02:00
|
|
|
* @var \Doctrine\ORM\UnitOfWork The UnitOfWork this factory uses to retrieve persisters
|
2012-12-13 10:28:55 +00:00
|
|
|
*/
|
2012-10-20 01:27:53 +02:00
|
|
|
private $uow;
|
2012-12-13 10:28:55 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @var string
|
|
|
|
*/
|
2012-10-20 01:27:53 +02:00
|
|
|
private $proxyNs;
|
2009-07-16 13:29:15 +00:00
|
|
|
|
2015-01-08 15:59:17 +01:00
|
|
|
/**
|
|
|
|
* The IdentifierFlattener used for manipulating identifiers
|
|
|
|
*
|
|
|
|
* @var \Doctrine\ORM\Utility\IdentifierFlattener
|
|
|
|
*/
|
|
|
|
private $identifierFlattener;
|
|
|
|
|
2009-07-16 13:29:15 +00:00
|
|
|
/**
|
2010-04-26 13:02:30 +02:00
|
|
|
* Initializes a new instance of the <tt>ProxyFactory</tt> class that is
|
|
|
|
* connected to the given <tt>EntityManager</tt>.
|
|
|
|
*
|
2014-12-05 14:02:47 +01:00
|
|
|
* @param EntityManagerInterface $em The EntityManager the new factory works for.
|
|
|
|
* @param string $proxyDir The directory to use for the proxy classes. It must exist.
|
|
|
|
* @param string $proxyNs The namespace to use for the proxy classes.
|
|
|
|
* @param boolean|int $autoGenerate The strategy for automatically generating proxy classes. Possible
|
|
|
|
* values are constants of Doctrine\Common\Proxy\AbstractProxyFactory.
|
2009-07-16 13:29:15 +00:00
|
|
|
*/
|
2014-12-05 14:02:47 +01:00
|
|
|
public function __construct(EntityManagerInterface $em, $proxyDir, $proxyNs, $autoGenerate = AbstractProxyFactory::AUTOGENERATE_NEVER)
|
2009-07-16 13:29:15 +00:00
|
|
|
{
|
2013-01-06 15:11:30 +01:00
|
|
|
$proxyGenerator = new ProxyGenerator($proxyDir, $proxyNs);
|
2010-03-05 16:35:00 +00:00
|
|
|
|
2013-01-06 15:11:30 +01:00
|
|
|
$proxyGenerator->setPlaceholder('baseProxyInterface', 'Doctrine\ORM\Proxy\Proxy');
|
|
|
|
parent::__construct($proxyGenerator, $em->getMetadataFactory(), $autoGenerate);
|
2010-03-05 16:35:00 +00:00
|
|
|
|
2015-01-08 15:59:17 +01:00
|
|
|
$this->em = $em;
|
|
|
|
$this->uow = $em->getUnitOfWork();
|
|
|
|
$this->proxyNs = $proxyNs;
|
|
|
|
$this->identifierFlattener = new IdentifierFlattener($this->uow, $em->getMetadataFactory());
|
2009-10-14 20:18:36 +00:00
|
|
|
}
|
2010-03-05 16:35:00 +00:00
|
|
|
|
2009-10-14 20:18:36 +00:00
|
|
|
/**
|
2013-01-06 15:11:30 +01:00
|
|
|
* {@inheritDoc}
|
2009-10-14 20:18:36 +00:00
|
|
|
*/
|
2013-01-06 15:11:30 +01:00
|
|
|
protected function skipClass(ClassMetadata $metadata)
|
2009-10-14 20:18:36 +00:00
|
|
|
{
|
2013-01-06 15:11:30 +01:00
|
|
|
/* @var $metadata \Doctrine\ORM\Mapping\ClassMetadataInfo */
|
|
|
|
return $metadata->isMappedSuperclass || $metadata->getReflectionClass()->isAbstract();
|
2012-10-20 01:27:53 +02:00
|
|
|
}
|
2011-10-24 19:45:23 -07:00
|
|
|
|
2012-10-20 01:27:53 +02:00
|
|
|
/**
|
2013-01-06 15:11:30 +01:00
|
|
|
* {@inheritDoc}
|
2012-10-20 01:27:53 +02:00
|
|
|
*/
|
2013-01-06 15:11:30 +01:00
|
|
|
protected function createProxyDefinition($className)
|
2012-10-20 01:27:53 +02:00
|
|
|
{
|
2013-01-06 15:11:30 +01:00
|
|
|
$classMetadata = $this->em->getClassMetadata($className);
|
|
|
|
$entityPersister = $this->uow->getEntityPersister($className);
|
2011-10-20 09:35:41 -07:00
|
|
|
|
2013-01-06 15:11:30 +01:00
|
|
|
return new ProxyDefinition(
|
|
|
|
ClassUtils::generateProxyClassName($className, $this->proxyNs),
|
|
|
|
$classMetadata->getIdentifierFieldNames(),
|
|
|
|
$classMetadata->getReflectionProperties(),
|
|
|
|
$this->createInitializer($classMetadata, $entityPersister),
|
|
|
|
$this->createCloner($classMetadata, $entityPersister)
|
|
|
|
);
|
2009-10-14 20:18:36 +00:00
|
|
|
}
|
2010-03-05 16:35:00 +00:00
|
|
|
|
2009-10-14 20:18:36 +00:00
|
|
|
/**
|
2013-01-06 15:11:30 +01:00
|
|
|
* Creates a closure capable of initializing a proxy
|
|
|
|
*
|
|
|
|
* @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $classMetadata
|
2015-01-15 05:01:52 +00:00
|
|
|
* @param \Doctrine\ORM\Persisters\Entity\EntityPersister $entityPersister
|
2013-01-06 15:11:30 +01:00
|
|
|
*
|
|
|
|
* @return \Closure
|
|
|
|
*
|
|
|
|
* @throws \Doctrine\ORM\EntityNotFoundException
|
2009-10-14 20:18:36 +00:00
|
|
|
*/
|
2013-02-13 20:42:13 -02:00
|
|
|
private function createInitializer(ClassMetadata $classMetadata, EntityPersister $entityPersister)
|
2009-10-14 20:18:36 +00:00
|
|
|
{
|
2012-10-20 01:27:53 +02:00
|
|
|
if ($classMetadata->getReflectionClass()->hasMethod('__wakeup')) {
|
2015-01-14 20:12:27 +01:00
|
|
|
return function (BaseProxy $proxy) use ($entityPersister, $classMetadata) {
|
2013-05-09 21:14:58 +02:00
|
|
|
$initializer = $proxy->__getInitializer();
|
|
|
|
$cloner = $proxy->__getCloner();
|
|
|
|
|
2012-10-20 01:27:53 +02:00
|
|
|
$proxy->__setInitializer(null);
|
|
|
|
$proxy->__setCloner(null);
|
2010-03-05 16:35:00 +00:00
|
|
|
|
2012-10-20 01:27:53 +02:00
|
|
|
if ($proxy->__isInitialized()) {
|
|
|
|
return;
|
2009-10-14 20:18:36 +00:00
|
|
|
}
|
2010-03-05 16:35:00 +00:00
|
|
|
|
2012-10-20 01:27:53 +02:00
|
|
|
$properties = $proxy->__getLazyProperties();
|
2011-11-08 09:43:06 +01:00
|
|
|
|
2012-10-20 01:27:53 +02:00
|
|
|
foreach ($properties as $propertyName => $property) {
|
2013-02-13 20:42:13 -02:00
|
|
|
if ( ! isset($proxy->$propertyName)) {
|
2012-10-20 01:27:53 +02:00
|
|
|
$proxy->$propertyName = $properties[$propertyName];
|
|
|
|
}
|
2011-05-20 20:50:03 +02:00
|
|
|
}
|
2010-03-05 16:35:00 +00:00
|
|
|
|
2012-10-20 01:27:53 +02:00
|
|
|
$proxy->__setInitialized(true);
|
|
|
|
$proxy->__wakeup();
|
2012-01-16 10:57:29 +01:00
|
|
|
|
2015-01-13 02:48:57 +01:00
|
|
|
$identifier = $classMetadata->getIdentifierValues($proxy);
|
|
|
|
|
|
|
|
if (null === $entityPersister->loadById($identifier, $proxy)) {
|
2013-05-09 21:14:58 +02:00
|
|
|
$proxy->__setInitializer($initializer);
|
|
|
|
$proxy->__setCloner($cloner);
|
|
|
|
$proxy->__setInitialized(false);
|
|
|
|
|
2015-01-13 02:51:47 +01:00
|
|
|
throw EntityNotFoundException::fromClassNameAndIdentifier(
|
2015-01-13 02:48:57 +01:00
|
|
|
$classMetadata->getName(),
|
2015-01-14 20:12:27 +01:00
|
|
|
$this->identifierFlattener->flattenIdentifier($classMetadata, $identifier)
|
2015-01-13 02:48:57 +01:00
|
|
|
);
|
2012-10-20 01:27:53 +02:00
|
|
|
}
|
|
|
|
};
|
2013-01-06 15:11:30 +01:00
|
|
|
}
|
2012-01-16 10:57:29 +01:00
|
|
|
|
2015-01-14 20:12:27 +01:00
|
|
|
return function (BaseProxy $proxy) use ($entityPersister, $classMetadata) {
|
2013-05-09 21:14:58 +02:00
|
|
|
$initializer = $proxy->__getInitializer();
|
|
|
|
$cloner = $proxy->__getCloner();
|
|
|
|
|
2013-01-06 15:11:30 +01:00
|
|
|
$proxy->__setInitializer(null);
|
|
|
|
$proxy->__setCloner(null);
|
2012-01-16 10:57:29 +01:00
|
|
|
|
2013-01-06 15:11:30 +01:00
|
|
|
if ($proxy->__isInitialized()) {
|
|
|
|
return;
|
|
|
|
}
|
2011-05-20 20:50:03 +02:00
|
|
|
|
2013-01-06 15:11:30 +01:00
|
|
|
$properties = $proxy->__getLazyProperties();
|
|
|
|
|
|
|
|
foreach ($properties as $propertyName => $property) {
|
|
|
|
if (!isset($proxy->$propertyName)) {
|
|
|
|
$proxy->$propertyName = $properties[$propertyName];
|
2009-10-14 20:18:36 +00:00
|
|
|
}
|
2013-01-06 15:11:30 +01:00
|
|
|
}
|
2010-03-05 16:35:00 +00:00
|
|
|
|
2013-01-06 15:11:30 +01:00
|
|
|
$proxy->__setInitialized(true);
|
2010-03-05 16:35:00 +00:00
|
|
|
|
2015-01-13 02:48:57 +01:00
|
|
|
$identifier = $classMetadata->getIdentifierValues($proxy);
|
|
|
|
|
|
|
|
if (null === $entityPersister->loadById($identifier, $proxy)) {
|
2013-05-09 21:14:58 +02:00
|
|
|
$proxy->__setInitializer($initializer);
|
|
|
|
$proxy->__setCloner($cloner);
|
|
|
|
$proxy->__setInitialized(false);
|
|
|
|
|
2015-01-13 02:51:47 +01:00
|
|
|
throw EntityNotFoundException::fromClassNameAndIdentifier(
|
2015-01-13 02:48:57 +01:00
|
|
|
$classMetadata->getName(),
|
2015-01-14 20:12:27 +01:00
|
|
|
$this->identifierFlattener->flattenIdentifier($classMetadata, $identifier)
|
2015-01-13 02:48:57 +01:00
|
|
|
);
|
2013-01-06 15:11:30 +01:00
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2010-03-05 16:35:00 +00:00
|
|
|
|
2013-01-06 15:11:30 +01:00
|
|
|
/**
|
|
|
|
* Creates a closure capable of finalizing state a cloned proxy
|
|
|
|
*
|
|
|
|
* @param \Doctrine\Common\Persistence\Mapping\ClassMetadata $classMetadata
|
2015-01-15 05:01:52 +00:00
|
|
|
* @param \Doctrine\ORM\Persisters\Entity\EntityPersister $entityPersister
|
2013-01-06 15:11:30 +01:00
|
|
|
*
|
|
|
|
* @return \Closure
|
|
|
|
*
|
|
|
|
* @throws \Doctrine\ORM\EntityNotFoundException
|
|
|
|
*/
|
2013-02-13 20:42:13 -02:00
|
|
|
private function createCloner(ClassMetadata $classMetadata, EntityPersister $entityPersister)
|
2013-01-06 15:11:30 +01:00
|
|
|
{
|
2015-01-14 20:12:27 +01:00
|
|
|
return function (BaseProxy $proxy) use ($entityPersister, $classMetadata) {
|
2012-10-20 01:27:53 +02:00
|
|
|
if ($proxy->__isInitialized()) {
|
|
|
|
return;
|
2011-07-09 12:12:44 +02:00
|
|
|
}
|
|
|
|
|
2012-10-20 01:27:53 +02:00
|
|
|
$proxy->__setInitialized(true);
|
|
|
|
$proxy->__setInitializer(null);
|
2015-01-08 15:59:17 +01:00
|
|
|
|
2015-01-13 02:48:57 +01:00
|
|
|
$class = $entityPersister->getClassMetadata();
|
|
|
|
$identifier = $classMetadata->getIdentifierValues($proxy);
|
|
|
|
$original = $entityPersister->loadById($identifier);
|
2011-08-19 06:11:58 +03:00
|
|
|
|
2012-10-20 01:27:53 +02:00
|
|
|
if (null === $original) {
|
2015-01-13 02:51:47 +01:00
|
|
|
throw EntityNotFoundException::fromClassNameAndIdentifier(
|
2015-01-13 02:48:57 +01:00
|
|
|
$classMetadata->getName(),
|
2015-01-14 20:12:27 +01:00
|
|
|
$this->identifierFlattener->flattenIdentifier($classMetadata, $identifier)
|
2015-01-13 02:48:57 +01:00
|
|
|
);
|
2012-10-20 01:27:53 +02:00
|
|
|
}
|
2012-01-16 10:57:29 +01:00
|
|
|
|
2013-02-13 20:42:13 -02:00
|
|
|
foreach ($class->getReflectionClass()->getProperties() as $property) {
|
|
|
|
if ( ! $class->hasField($property->name) && ! $class->hasAssociation($property->name)) {
|
|
|
|
continue;
|
2012-10-20 01:27:53 +02:00
|
|
|
}
|
2013-02-13 20:42:13 -02:00
|
|
|
|
|
|
|
$property->setAccessible(true);
|
|
|
|
$property->setValue($proxy, $property->getValue($original));
|
2011-02-26 12:47:59 +01:00
|
|
|
}
|
2012-10-20 01:27:53 +02:00
|
|
|
};
|
2011-02-26 12:47:59 +01:00
|
|
|
}
|
2009-07-16 13:29:15 +00:00
|
|
|
}
|