' . $valueHolder->getName(); } if ($interfaceName !== null) { return '$targetObject = ' . $target . ';' . "\n\n" . self::getUndefinedPropertyNotice($operationType, $nameParameter, $interfaceName) . self::getOperation($operationType, $nameParameter, $valueParameter); } return '$realInstanceReflection = new \\ReflectionClass(get_parent_class($this));' . "\n\n" . 'if (! $realInstanceReflection->hasProperty($' . $nameParameter . ')) {' . "\n" . ' $targetObject = ' . $target . ';' . "\n\n" . self::getUndefinedPropertyNotice($operationType, $nameParameter) . ' ' . self::getOperation($operationType, $nameParameter, $valueParameter) . "\n" . " return;\n" . '}' . "\n\n" . '$targetObject = ' . self::getTargetObject($valueHolder) . ";\n" . '$accessor = function ' . $byRef . '() use ($targetObject, $' . $nameParameter . $value . ') {' . "\n" . ' ' . self::getOperation($operationType, $nameParameter, $valueParameter) . "\n" . "};\n" . self::getScopeReBind() . ( $returnPropertyName ? '$' . $returnPropertyName . ' = ' . $byRef . '$accessor();' : '$returnValue = ' . $byRef . '$accessor();' . "\n\n" . 'return $returnValue;' ); } /** * This will generate code that triggers a notice if access is attempted on a non-existing property */ private static function getUndefinedPropertyNotice(string $operationType, string $nameParameter, ?string $interfaceName = null): string { if ($operationType !== self::OPERATION_GET) { return ''; } $code = ' $backtrace = debug_backtrace(false, 1);' . "\n" . ' trigger_error(' . "\n" . ' sprintf(' . "\n" . ' \'Undefined property: %s::$%s in %s on line %s\',' . "\n" . ' ' . ($interfaceName !== null ? var_export($interfaceName, true) : 'get_parent_class($this)') . ',' . "\n" . ' $' . $nameParameter . ',' . "\n" . ' $backtrace[0][\'file\'],' . "\n" . ' $backtrace[0][\'line\']' . "\n" . ' ),' . "\n" . ' \E_USER_NOTICE' . "\n" . ' );' . "\n"; if ($interfaceName !== null) { $code = str_replace("\n ", "\n", substr($code, 4)); } return $code; } /** * Defines whether the given operation produces a reference. * * Note: if the object is a wrapper, the wrapped instance is accessed directly. If the object * is a ghost or the proxy has no wrapper, then an instance of the parent class is created via * on-the-fly unserialization */ private static function getByRefReturnValue(string $operationType): string { return $operationType === self::OPERATION_GET || $operationType === self::OPERATION_SET ? '& ' : ''; } /** * Retrieves the logic to fetch the object on which access should be attempted */ private static function getTargetObject(?PropertyGenerator $valueHolder = null): string { if ($valueHolder) { return '$this->' . $valueHolder->getName(); } return '$realInstanceReflection->newInstanceWithoutConstructor()'; } /** * @throws InvalidArgumentException */ private static function getOperation(string $operationType, string $nameParameter, ?string $valueParameter): string { switch ($operationType) { case self::OPERATION_GET: return 'return $targetObject->$' . $nameParameter . ';'; case self::OPERATION_SET: if ($valueParameter === null) { throw new InvalidArgumentException('Parameter $valueParameter not provided'); } return '$targetObject->$' . $nameParameter . ' = $' . $valueParameter . '; return $targetObject->$' . $nameParameter . ';'; case self::OPERATION_ISSET: return 'return isset($targetObject->$' . $nameParameter . ');'; case self::OPERATION_UNSET: return 'unset($targetObject->$' . $nameParameter . ');'; } throw new InvalidArgumentException(sprintf('Invalid operation "%s" provided', $operationType)); } /** * Generates code to bind operations to the parent scope */ private static function getScopeReBind(): string { return '$backtrace = debug_backtrace(true, 2);' . "\n" . '$scopeObject = isset($backtrace[1][\'object\'])' . ' ? $backtrace[1][\'object\'] : new \ProxyManager\Stub\EmptyClassStub();' . "\n" . '$accessor = $accessor->bindTo($scopeObject, get_class($scopeObject));' . "\n"; } }