* @license MIT * * @group Functional * @coversNothing */ class LazyLoadingValueHolderFunctionalTest extends PHPUnit_Framework_TestCase { /** * @dataProvider getProxyMethods */ public function testMethodCalls($className, $instance, $method, $params, $expectedValue) { $proxyName = $this->generateProxy($className); /* @var $proxy \ProxyManager\Proxy\VirtualProxyInterface|BaseClass */ $proxy = new $proxyName($this->createInitializer($className, $instance)); $this->assertFalse($proxy->isProxyInitialized()); $this->assertSame($expectedValue, call_user_func_array(array($proxy, $method), $params)); $this->assertTrue($proxy->isProxyInitialized()); $this->assertSame($instance, $proxy->getWrappedValueHolderValue()); } /** * @dataProvider getProxyMethods */ public function testMethodCallsAfterUnSerialization($className, $instance, $method, $params, $expectedValue) { $proxyName = $this->generateProxy($className); /* @var $proxy \ProxyManager\Proxy\VirtualProxyInterface|BaseClass */ $proxy = unserialize(serialize(new $proxyName($this->createInitializer($className, $instance)))); $this->assertTrue($proxy->isProxyInitialized()); $this->assertSame($expectedValue, call_user_func_array(array($proxy, $method), $params)); $this->assertEquals($instance, $proxy->getWrappedValueHolderValue()); } /** * @dataProvider getProxyMethods */ public function testMethodCallsAfterCloning($className, $instance, $method, $params, $expectedValue) { $proxyName = $this->generateProxy($className); /* @var $proxy \ProxyManager\Proxy\VirtualProxyInterface|BaseClass */ $proxy = new $proxyName($this->createInitializer($className, $instance)); $cloned = clone $proxy; $this->assertTrue($cloned->isProxyInitialized()); $this->assertNotSame($proxy->getWrappedValueHolderValue(), $cloned->getWrappedValueHolderValue()); $this->assertSame($expectedValue, call_user_func_array(array($cloned, $method), $params)); $this->assertEquals($instance, $cloned->getWrappedValueHolderValue()); } /** * @dataProvider getPropertyAccessProxies */ public function testPropertyReadAccess($instance, $proxy, $publicProperty, $propertyValue) { /* @var $proxy \ProxyManager\Proxy\VirtualProxyInterface|BaseClass */ $this->assertSame($propertyValue, $proxy->$publicProperty); $this->assertTrue($proxy->isProxyInitialized()); $this->assertEquals($instance, $proxy->getWrappedValueHolderValue()); } /** * @dataProvider getPropertyAccessProxies */ public function testPropertyWriteAccess($instance, $proxy, $publicProperty) { /* @var $proxy \ProxyManager\Proxy\VirtualProxyInterface|BaseClass */ $newValue = uniqid(); $proxy->$publicProperty = $newValue; $this->assertTrue($proxy->isProxyInitialized()); $this->assertSame($newValue, $proxy->$publicProperty); $this->assertSame($newValue, $proxy->getWrappedValueHolderValue()->$publicProperty); } /** * @dataProvider getPropertyAccessProxies */ public function testPropertyExistence($instance, $proxy, $publicProperty) { /* @var $proxy \ProxyManager\Proxy\VirtualProxyInterface|BaseClass */ $this->assertSame(isset($instance->$publicProperty), isset($proxy->$publicProperty)); $this->assertTrue($proxy->isProxyInitialized()); $this->assertEquals($instance, $proxy->getWrappedValueHolderValue()); } /** * @dataProvider getPropertyAccessProxies */ public function testPropertyAbsence($instance, $proxy, $publicProperty) { /* @var $proxy \ProxyManager\Proxy\VirtualProxyInterface|BaseClass */ $instance = $proxy->getWrappedValueHolderValue() ? $proxy->getWrappedValueHolderValue() : $instance; $instance->$publicProperty = null; $this->assertFalse(isset($proxy->$publicProperty)); $this->assertTrue($proxy->isProxyInitialized()); } /** * @dataProvider getPropertyAccessProxies */ public function testPropertyUnset($instance, $proxy, $publicProperty) { /* @var $proxy \ProxyManager\Proxy\VirtualProxyInterface|BaseClass */ $instance = $proxy->getWrappedValueHolderValue() ? $proxy->getWrappedValueHolderValue() : $instance; unset($proxy->$publicProperty); $this->assertTrue($proxy->isProxyInitialized()); $this->assertFalse(isset($instance->$publicProperty)); $this->assertFalse(isset($proxy->$publicProperty)); } /** * Verifies that accessing a public property containing an array behaves like in a normal context */ public function testCanWriteToArrayKeysInPublicProperty() { $instance = new ClassWithPublicArrayProperty(); $className = get_class($instance); $initializer = $this->createInitializer($className, $instance); $proxyName = $this->generateProxy($className); /* @var $proxy ClassWithPublicArrayProperty */ $proxy = new $proxyName($initializer); $proxy->arrayProperty['foo'] = 'bar'; $this->assertSame('bar', $proxy->arrayProperty['foo']); $proxy->arrayProperty = array('tab' => 'taz'); $this->assertSame(array('tab' => 'taz'), $proxy->arrayProperty); } /** * Verifies that public properties retrieved via `__get` don't get modified in the object itself */ public function testWillNotModifyRetrievedPublicProperties() { $instance = new ClassWithPublicProperties(); $className = get_class($instance); $initializer = $this->createInitializer($className, $instance); $proxyName = $this->generateProxy($className); /* @var $proxy ClassWithPublicProperties */ $proxy = new $proxyName($initializer); $variable = $proxy->property0; $this->assertSame('property0', $variable); $variable = 'foo'; $this->assertSame('property0', $proxy->property0); } /** * Verifies that public properties references retrieved via `__get` modify in the object state */ public function testWillModifyByRefRetrievedPublicProperties() { $instance = new ClassWithPublicProperties(); $className = get_class($instance); $initializer = $this->createInitializer($className, $instance); $proxyName = $this->generateProxy($className); /* @var $proxy ClassWithPublicProperties */ $proxy = new $proxyName($initializer); $variable = & $proxy->property0; $this->assertSame('property0', $variable); $variable = 'foo'; $this->assertSame('foo', $proxy->property0); } /** * @group 16 * * Verifies that initialization of a value holder proxy may happen multiple times */ public function testWillAllowMultipleProxyInitialization() { $proxyClass = $this->generateProxy('ProxyManagerTestAsset\\BaseClass'); $counter = 0; $initializer = function (& $wrappedInstance) use (& $counter) { $wrappedInstance = new BaseClass(); $wrappedInstance->publicProperty = (string) ($counter += 1); }; /* @var $proxy BaseClass */ $proxy = new $proxyClass($initializer); $this->assertSame('1', $proxy->publicProperty); $this->assertSame('2', $proxy->publicProperty); $this->assertSame('3', $proxy->publicProperty); } /** * Generates a proxy for the given class name, and retrieves its class name * * @param string $parentClassName * * @return string */ private function generateProxy($parentClassName) { $generatedClassName = __NAMESPACE__ . '\\' . UniqueIdentifierGenerator::getIdentifier('Foo'); $generator = new LazyLoadingValueHolderGenerator(); $generatedClass = new ClassGenerator($generatedClassName); $strategy = new EvaluatingGeneratorStrategy(); $generator->generate(new ReflectionClass($parentClassName), $generatedClass); $strategy->generate($generatedClass); return $generatedClassName; } /** * @param string $className * @param object $realInstance * @param Mock $initializerMatcher * * @return \Closure */ private function createInitializer($className, $realInstance, Mock $initializerMatcher = null) { if (null === $initializerMatcher) { $initializerMatcher = $this->getMock('stdClass', array('__invoke')); $initializerMatcher ->expects($this->once()) ->method('__invoke') ->with( $this->logicalAnd( $this->isInstanceOf('ProxyManager\\Proxy\\VirtualProxyInterface'), $this->isInstanceOf($className) ), $realInstance ); } $initializerMatcher = $initializerMatcher ?: $this->getMock('stdClass', array('__invoke')); return function ( & $wrappedObject, VirtualProxyInterface $proxy, $method, $params, & $initializer ) use ( $initializerMatcher, $realInstance ) { $initializer = null; $wrappedObject = $realInstance; $initializerMatcher->__invoke($proxy, $wrappedObject, $method, $params); }; } /** * Generates a list of object | invoked method | parameters | expected result * * @return array */ public function getProxyMethods() { $selfHintParam = new ClassWithSelfHint(); $data = array( array( 'ProxyManagerTestAsset\\BaseClass', new BaseClass(), 'publicMethod', array(), 'publicMethodDefault' ), array( 'ProxyManagerTestAsset\\BaseClass', new BaseClass(), 'publicTypeHintedMethod', array(new \stdClass()), 'publicTypeHintedMethodDefault' ), array( 'ProxyManagerTestAsset\\BaseClass', new BaseClass(), 'publicByReferenceMethod', array(), 'publicByReferenceMethodDefault' ), array( 'ProxyManagerTestAsset\\BaseInterface', new BaseClass(), 'publicMethod', array(), 'publicMethodDefault' ), ); if (PHP_VERSION_ID >= 50401) { // PHP < 5.4.1 misbehaves, throwing strict standards, see https://bugs.php.net/bug.php?id=60573 $data[] = array( 'ProxyManagerTestAsset\\ClassWithSelfHint', new ClassWithSelfHint(), 'selfHintMethod', array('parameter' => $selfHintParam), $selfHintParam ); } return $data; } /** * Generates proxies and instances with a public property to feed to the property accessor methods * * @return array */ public function getPropertyAccessProxies() { $instance1 = new BaseClass(); $proxyName1 = $this->generateProxy(get_class($instance1)); $instance2 = new BaseClass(); $proxyName2 = $this->generateProxy(get_class($instance2)); return array( array( $instance1, new $proxyName1($this->createInitializer('ProxyManagerTestAsset\\BaseClass', $instance1)), 'publicProperty', 'publicPropertyDefault', ), array( $instance2, unserialize( serialize(new $proxyName2($this->createInitializer('ProxyManagerTestAsset\\BaseClass', $instance2))) ), 'publicProperty', 'publicPropertyDefault', ), ); } }