, skipDestructor?: bool} $proxyOptions * * @return void * * @throws InvalidProxiedClassException * @throws InvalidArgumentException */ public function generate(ReflectionClass $originalClass, ClassGenerator $classGenerator, array $proxyOptions = []) { CanProxyAssertion::assertClassCanBeProxied($originalClass, false); $filteredProperties = Properties::fromReflectionClass($originalClass) ->filter($proxyOptions['skippedProperties'] ?? []); $publicProperties = new PublicPropertiesMap($filteredProperties, true); $privateProperties = new PrivatePropertiesMap($filteredProperties); $protectedProperties = new ProtectedPropertiesMap($filteredProperties); $skipDestructor = ($proxyOptions['skipDestructor'] ?? false) && $originalClass->hasMethod('__destruct'); $classGenerator->setExtendedClass($originalClass->getName()); $classGenerator->setImplementedInterfaces([GhostObjectInterface::class]); $classGenerator->addPropertyFromGenerator($initializer = new InitializerProperty()); $classGenerator->addPropertyFromGenerator($initializationTracker = new InitializationTracker()); $classGenerator->addPropertyFromGenerator($publicProperties); $classGenerator->addPropertyFromGenerator($privateProperties); $classGenerator->addPropertyFromGenerator($protectedProperties); $init = new CallInitializer($initializer, $initializationTracker, $filteredProperties); array_map( static function (MethodGenerator $generatedMethod) use ($originalClass, $classGenerator): void { ClassGeneratorUtils::addMethodIfNotFinal($originalClass, $classGenerator, $generatedMethod); }, array_merge( $this->getAbstractProxiedMethods($originalClass, $skipDestructor), [ $init, new StaticProxyConstructor($initializer, $filteredProperties), new MagicGet( $originalClass, $initializer, $init, $publicProperties, $protectedProperties, $privateProperties, $initializationTracker ), new MagicSet( $originalClass, $initializer, $init, $publicProperties, $protectedProperties, $privateProperties ), new MagicIsset( $originalClass, $initializer, $init, $publicProperties, $protectedProperties, $privateProperties ), new MagicUnset( $originalClass, $initializer, $init, $publicProperties, $protectedProperties, $privateProperties ), new MagicClone($originalClass, $initializer, $init), new MagicSleep($originalClass, $initializer, $init), new SetProxyInitializer($initializer), new GetProxyInitializer($initializer), new InitializeProxy($initializer, $init), new IsProxyInitialized($initializer), ], $skipDestructor ? [new SkipDestructor($initializer)] : [] ) ); } /** * Retrieves all abstract methods to be proxied * * @return MethodGenerator[] */ private function getAbstractProxiedMethods(ReflectionClass $originalClass, bool $skipDestructor): array { $excludedMethods = ProxiedMethodsFilter::DEFAULT_EXCLUDED; if ($skipDestructor) { $excludedMethods[] = '__destruct'; } return array_map( static function (ReflectionMethod $method): ProxyManagerMethodGenerator { $generated = ProxyManagerMethodGenerator::fromReflectionWithoutBodyAndDocBlock( new MethodReflection($method->getDeclaringClass()->getName(), $method->getName()) ); $generated->setAbstract(false); return $generated; }, ProxiedMethodsFilter::getAbstractProxiedMethods($originalClass, $excludedMethods) ); } }