123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262 |
- <?php
- namespace Doctrine\Instantiator;
- use ArrayIterator;
- use Doctrine\Instantiator\Exception\ExceptionInterface;
- use Doctrine\Instantiator\Exception\InvalidArgumentException;
- use Doctrine\Instantiator\Exception\UnexpectedValueException;
- use Exception;
- use ReflectionClass;
- use ReflectionException;
- use Serializable;
- use function class_exists;
- use function enum_exists;
- use function is_subclass_of;
- use function restore_error_handler;
- use function set_error_handler;
- use function sprintf;
- use function strlen;
- use function unserialize;
- use const PHP_VERSION_ID;
- final class Instantiator implements InstantiatorInterface
- {
- /**
- * Markers used internally by PHP to define whether {@see \unserialize} should invoke
- * the method {@see \Serializable::unserialize()} when dealing with classes implementing
- * the {@see \Serializable} interface.
- *
- * @deprecated This constant will be private in 2.0
- */
- public const SERIALIZATION_FORMAT_USE_UNSERIALIZER = 'C';
- /** @deprecated This constant will be private in 2.0 */
- public const SERIALIZATION_FORMAT_AVOID_UNSERIALIZER = 'O';
- /**
- * Used to instantiate specific classes, indexed by class name.
- *
- * @var callable[]
- */
- private static $cachedInstantiators = [];
- /**
- * Array of objects that can directly be cloned, indexed by class name.
- *
- * @var object[]
- */
- private static $cachedCloneables = [];
- /**
- * @param string $className
- * @phpstan-param class-string<T> $className
- *
- * @return object
- * @phpstan-return T
- *
- * @throws ExceptionInterface
- *
- * @template T of object
- */
- public function instantiate($className)
- {
- if (isset(self::$cachedCloneables[$className])) {
- /** @phpstan-var T */
- $cachedCloneable = self::$cachedCloneables[$className];
- return clone $cachedCloneable;
- }
- if (isset(self::$cachedInstantiators[$className])) {
- $factory = self::$cachedInstantiators[$className];
- return $factory();
- }
- return $this->buildAndCacheFromFactory($className);
- }
- /**
- * Builds the requested object and caches it in static properties for performance
- *
- * @phpstan-param class-string<T> $className
- *
- * @return object
- * @phpstan-return T
- *
- * @template T of object
- */
- private function buildAndCacheFromFactory(string $className)
- {
- $factory = self::$cachedInstantiators[$className] = $this->buildFactory($className);
- $instance = $factory();
- if ($this->isSafeToClone(new ReflectionClass($instance))) {
- self::$cachedCloneables[$className] = clone $instance;
- }
- return $instance;
- }
- /**
- * Builds a callable capable of instantiating the given $className without
- * invoking its constructor.
- *
- * @phpstan-param class-string<T> $className
- *
- * @phpstan-return callable(): T
- *
- * @throws InvalidArgumentException
- * @throws UnexpectedValueException
- * @throws ReflectionException
- *
- * @template T of object
- */
- private function buildFactory(string $className): callable
- {
- $reflectionClass = $this->getReflectionClass($className);
- if ($this->isInstantiableViaReflection($reflectionClass)) {
- return [$reflectionClass, 'newInstanceWithoutConstructor'];
- }
- $serializedString = sprintf(
- '%s:%d:"%s":0:{}',
- is_subclass_of($className, Serializable::class) ? self::SERIALIZATION_FORMAT_USE_UNSERIALIZER : self::SERIALIZATION_FORMAT_AVOID_UNSERIALIZER,
- strlen($className),
- $className
- );
- $this->checkIfUnSerializationIsSupported($reflectionClass, $serializedString);
- return static function () use ($serializedString) {
- return unserialize($serializedString);
- };
- }
- /**
- * @phpstan-param class-string<T> $className
- *
- * @phpstan-return ReflectionClass<T>
- *
- * @throws InvalidArgumentException
- * @throws ReflectionException
- *
- * @template T of object
- */
- private function getReflectionClass(string $className): ReflectionClass
- {
- if (! class_exists($className)) {
- throw InvalidArgumentException::fromNonExistingClass($className);
- }
- if (PHP_VERSION_ID >= 80100 && enum_exists($className, false)) {
- throw InvalidArgumentException::fromEnum($className);
- }
- $reflection = new ReflectionClass($className);
- if ($reflection->isAbstract()) {
- throw InvalidArgumentException::fromAbstractClass($reflection);
- }
- return $reflection;
- }
- /**
- * @phpstan-param ReflectionClass<T> $reflectionClass
- *
- * @throws UnexpectedValueException
- *
- * @template T of object
- */
- private function checkIfUnSerializationIsSupported(ReflectionClass $reflectionClass, string $serializedString): void
- {
- set_error_handler(static function (int $code, string $message, string $file, int $line) use ($reflectionClass, &$error): bool {
- $error = UnexpectedValueException::fromUncleanUnSerialization(
- $reflectionClass,
- $message,
- $code,
- $file,
- $line
- );
- return true;
- });
- try {
- $this->attemptInstantiationViaUnSerialization($reflectionClass, $serializedString);
- } finally {
- restore_error_handler();
- }
- if ($error) {
- throw $error;
- }
- }
- /**
- * @phpstan-param ReflectionClass<T> $reflectionClass
- *
- * @throws UnexpectedValueException
- *
- * @template T of object
- */
- private function attemptInstantiationViaUnSerialization(ReflectionClass $reflectionClass, string $serializedString): void
- {
- try {
- unserialize($serializedString);
- } catch (Exception $exception) {
- throw UnexpectedValueException::fromSerializationTriggeredException($reflectionClass, $exception);
- }
- }
- /**
- * @phpstan-param ReflectionClass<T> $reflectionClass
- *
- * @template T of object
- */
- private function isInstantiableViaReflection(ReflectionClass $reflectionClass): bool
- {
- return ! ($this->hasInternalAncestors($reflectionClass) && $reflectionClass->isFinal());
- }
- /**
- * Verifies whether the given class is to be considered internal
- *
- * @phpstan-param ReflectionClass<T> $reflectionClass
- *
- * @template T of object
- */
- private function hasInternalAncestors(ReflectionClass $reflectionClass): bool
- {
- do {
- if ($reflectionClass->isInternal()) {
- return true;
- }
- $reflectionClass = $reflectionClass->getParentClass();
- } while ($reflectionClass);
- return false;
- }
- /**
- * Checks if a class is cloneable
- *
- * Classes implementing `__clone` cannot be safely cloned, as that may cause side-effects.
- *
- * @phpstan-param ReflectionClass<T> $reflectionClass
- *
- * @template T of object
- */
- private function isSafeToClone(ReflectionClass $reflectionClass): bool
- {
- return $reflectionClass->isCloneable()
- && ! $reflectionClass->hasMethod('__clone')
- && ! $reflectionClass->isSubclassOf(ArrayIterator::class);
- }
- }
|