* * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace Symfony\Component\DependencyInjection\Tests\Compiler; use PHPUnit\Framework\TestCase; use Symfony\Component\DependencyInjection\Compiler\AutowirePass; use Symfony\Component\DependencyInjection\ContainerBuilder; use Symfony\Component\DependencyInjection\Reference; /** * @author Kévin Dunglas */ class AutowirePassTest extends TestCase { public function testProcess() { $container = new ContainerBuilder(); $container->register('foo', __NAMESPACE__.'\Foo'); $barDefinition = $container->register('bar', __NAMESPACE__.'\Bar'); $barDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $this->assertCount(1, $container->getDefinition('bar')->getArguments()); $this->assertEquals('foo', (string) $container->getDefinition('bar')->getArgument(0)); } public function testProcessAutowireParent() { $container = new ContainerBuilder(); $container->register('b', __NAMESPACE__.'\B'); $cDefinition = $container->register('c', __NAMESPACE__.'\C'); $cDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $this->assertCount(1, $container->getDefinition('c')->getArguments()); $this->assertEquals('b', (string) $container->getDefinition('c')->getArgument(0)); } public function testProcessAutowireInterface() { $container = new ContainerBuilder(); $container->register('f', __NAMESPACE__.'\F'); $gDefinition = $container->register('g', __NAMESPACE__.'\G'); $gDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $this->assertCount(3, $container->getDefinition('g')->getArguments()); $this->assertEquals('f', (string) $container->getDefinition('g')->getArgument(0)); $this->assertEquals('f', (string) $container->getDefinition('g')->getArgument(1)); $this->assertEquals('f', (string) $container->getDefinition('g')->getArgument(2)); } public function testCompleteExistingDefinition() { $container = new ContainerBuilder(); $container->register('b', __NAMESPACE__.'\B'); $container->register('f', __NAMESPACE__.'\F'); $hDefinition = $container->register('h', __NAMESPACE__.'\H')->addArgument(new Reference('b')); $hDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $this->assertCount(2, $container->getDefinition('h')->getArguments()); $this->assertEquals('b', (string) $container->getDefinition('h')->getArgument(0)); $this->assertEquals('f', (string) $container->getDefinition('h')->getArgument(1)); } public function testCompleteExistingDefinitionWithNotDefinedArguments() { $container = new ContainerBuilder(); $container->register('b', __NAMESPACE__.'\B'); $container->register('f', __NAMESPACE__.'\F'); $hDefinition = $container->register('h', __NAMESPACE__.'\H')->addArgument('')->addArgument(''); $hDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $this->assertCount(2, $container->getDefinition('h')->getArguments()); $this->assertEquals('b', (string) $container->getDefinition('h')->getArgument(0)); $this->assertEquals('f', (string) $container->getDefinition('h')->getArgument(1)); } /** * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException * @expectedExceptionMessage Unable to autowire argument of type "Symfony\Component\DependencyInjection\Tests\Compiler\CollisionInterface" for the service "a". Multiple services exist for this interface (c1, c2, c3). */ public function testTypeCollision() { $container = new ContainerBuilder(); $container->register('c1', __NAMESPACE__.'\CollisionA'); $container->register('c2', __NAMESPACE__.'\CollisionB'); $container->register('c3', __NAMESPACE__.'\CollisionB'); $aDefinition = $container->register('a', __NAMESPACE__.'\CannotBeAutowired'); $aDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); } /** * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException * @expectedExceptionMessage Unable to autowire argument of type "Symfony\Component\DependencyInjection\Tests\Compiler\Foo" for the service "a". Multiple services exist for this class (a1, a2). */ public function testTypeNotGuessable() { $container = new ContainerBuilder(); $container->register('a1', __NAMESPACE__.'\Foo'); $container->register('a2', __NAMESPACE__.'\Foo'); $aDefinition = $container->register('a', __NAMESPACE__.'\NotGuessableArgument'); $aDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); } /** * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException * @expectedExceptionMessage Unable to autowire argument of type "Symfony\Component\DependencyInjection\Tests\Compiler\A" for the service "a". Multiple services exist for this class (a1, a2). */ public function testTypeNotGuessableWithSubclass() { $container = new ContainerBuilder(); $container->register('a1', __NAMESPACE__.'\B'); $container->register('a2', __NAMESPACE__.'\B'); $aDefinition = $container->register('a', __NAMESPACE__.'\NotGuessableArgumentForSubclass'); $aDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); } /** * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException * @expectedExceptionMessage Unable to autowire argument of type "Symfony\Component\DependencyInjection\Tests\Compiler\CollisionInterface" for the service "a". No services were found matching this interface and it cannot be auto-registered. */ public function testTypeNotGuessableNoServicesFound() { $container = new ContainerBuilder(); $aDefinition = $container->register('a', __NAMESPACE__.'\CannotBeAutowired'); $aDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); } public function testTypeNotGuessableWithTypeSet() { $container = new ContainerBuilder(); $container->register('a1', __NAMESPACE__.'\Foo'); $container->register('a2', __NAMESPACE__.'\Foo'); $container->register('a3', __NAMESPACE__.'\Foo')->addAutowiringType(__NAMESPACE__.'\Foo'); $aDefinition = $container->register('a', __NAMESPACE__.'\NotGuessableArgument'); $aDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $this->assertCount(1, $container->getDefinition('a')->getArguments()); $this->assertEquals('a3', (string) $container->getDefinition('a')->getArgument(0)); } public function testWithTypeSet() { $container = new ContainerBuilder(); $container->register('c1', __NAMESPACE__.'\CollisionA'); $container->register('c2', __NAMESPACE__.'\CollisionB')->addAutowiringType(__NAMESPACE__.'\CollisionInterface'); $aDefinition = $container->register('a', __NAMESPACE__.'\CannotBeAutowired'); $aDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $this->assertCount(1, $container->getDefinition('a')->getArguments()); $this->assertEquals('c2', (string) $container->getDefinition('a')->getArgument(0)); } public function testCreateDefinition() { $container = new ContainerBuilder(); $coopTilleulsDefinition = $container->register('coop_tilleuls', __NAMESPACE__.'\LesTilleuls'); $coopTilleulsDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $this->assertCount(2, $container->getDefinition('coop_tilleuls')->getArguments()); $this->assertEquals('autowired.symfony\component\dependencyinjection\tests\compiler\dunglas', $container->getDefinition('coop_tilleuls')->getArgument(0)); $this->assertEquals('autowired.symfony\component\dependencyinjection\tests\compiler\dunglas', $container->getDefinition('coop_tilleuls')->getArgument(1)); $dunglasDefinition = $container->getDefinition('autowired.Symfony\Component\DependencyInjection\Tests\Compiler\Dunglas'); $this->assertEquals(__NAMESPACE__.'\Dunglas', $dunglasDefinition->getClass()); $this->assertFalse($dunglasDefinition->isPublic()); $this->assertCount(1, $dunglasDefinition->getArguments()); $this->assertEquals('autowired.symfony\component\dependencyinjection\tests\compiler\lille', $dunglasDefinition->getArgument(0)); $lilleDefinition = $container->getDefinition('autowired.Symfony\Component\DependencyInjection\Tests\Compiler\Lille'); $this->assertEquals(__NAMESPACE__.'\Lille', $lilleDefinition->getClass()); } public function testResolveParameter() { $container = new ContainerBuilder(); $container->setParameter('class_name', __NAMESPACE__.'\Foo'); $container->register('foo', '%class_name%'); $barDefinition = $container->register('bar', __NAMESPACE__.'\Bar'); $barDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $this->assertEquals('foo', $container->getDefinition('bar')->getArgument(0)); } public function testOptionalParameter() { $container = new ContainerBuilder(); $container->register('a', __NAMESPACE__.'\A'); $container->register('foo', __NAMESPACE__.'\Foo'); $optDefinition = $container->register('opt', __NAMESPACE__.'\OptionalParameter'); $optDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $definition = $container->getDefinition('opt'); $this->assertNull($definition->getArgument(0)); $this->assertEquals('a', $definition->getArgument(1)); $this->assertEquals('foo', $definition->getArgument(2)); } public function testDontTriggerAutowiring() { $container = new ContainerBuilder(); $container->register('foo', __NAMESPACE__.'\Foo'); $container->register('bar', __NAMESPACE__.'\Bar'); $pass = new AutowirePass(); $pass->process($container); $this->assertCount(0, $container->getDefinition('bar')->getArguments()); } /** * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException * @expectedExceptionMessage Cannot autowire argument 2 for Symfony\Component\DependencyInjection\Tests\Compiler\BadTypeHintedArgument because the type-hinted class does not exist (Class Symfony\Component\DependencyInjection\Tests\Compiler\NotARealClass does not exist). */ public function testClassNotFoundThrowsException() { $container = new ContainerBuilder(); $aDefinition = $container->register('a', __NAMESPACE__.'\BadTypeHintedArgument'); $aDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); } /** * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException * @expectedExceptionMessage Cannot autowire argument 2 for Symfony\Component\DependencyInjection\Tests\Compiler\BadParentTypeHintedArgument because the type-hinted class does not exist (Class Symfony\Component\DependencyInjection\Tests\Compiler\OptionalServiceClass does not exist). */ public function testParentClassNotFoundThrowsException() { $container = new ContainerBuilder(); $aDefinition = $container->register('a', __NAMESPACE__.'\BadParentTypeHintedArgument'); $aDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); } public function testDontUseAbstractServices() { $container = new ContainerBuilder(); $container->register('abstract_foo', __NAMESPACE__.'\Foo')->setAbstract(true); $container->register('foo', __NAMESPACE__.'\Foo'); $container->register('bar', __NAMESPACE__.'\Bar')->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $arguments = $container->getDefinition('bar')->getArguments(); $this->assertSame('foo', (string) $arguments[0]); } public function testSomeSpecificArgumentsAreSet() { $container = new ContainerBuilder(); $container->register('foo', __NAMESPACE__.'\Foo'); $container->register('a', __NAMESPACE__.'\A'); $container->register('dunglas', __NAMESPACE__.'\Dunglas'); $container->register('multiple', __NAMESPACE__.'\MultipleArguments') ->setAutowired(true) // set the 2nd (index 1) argument only: autowire the first and third // args are: A, Foo, Dunglas ->setArguments(array( 1 => new Reference('foo'), )); $pass = new AutowirePass(); $pass->process($container); $definition = $container->getDefinition('multiple'); $this->assertEquals( array( new Reference('a'), new Reference('foo'), new Reference('dunglas'), ), $definition->getArguments() ); } /** * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException * @expectedExceptionMessage Unable to autowire argument index 1 ($foo) for the service "arg_no_type_hint". If this is an object, give it a type-hint. Otherwise, specify this argument's value explicitly. */ public function testScalarArgsCannotBeAutowired() { $container = new ContainerBuilder(); $container->register('a', __NAMESPACE__.'\A'); $container->register('dunglas', __NAMESPACE__.'\Dunglas'); $container->register('arg_no_type_hint', __NAMESPACE__.'\MultipleArguments') ->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $container->getDefinition('arg_no_type_hint'); } /** * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException * @expectedExceptionMessage Unable to autowire argument index 1 ($foo) for the service "not_really_optional_scalar". If this is an object, give it a type-hint. Otherwise, specify this argument's value explicitly. */ public function testOptionalScalarNotReallyOptionalThrowException() { $container = new ContainerBuilder(); $container->register('a', __NAMESPACE__.'\A'); $container->register('lille', __NAMESPACE__.'\Lille'); $container->register('not_really_optional_scalar', __NAMESPACE__.'\MultipleArgumentsOptionalScalarNotReallyOptional') ->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); } public function testOptionalScalarArgsDontMessUpOrder() { $container = new ContainerBuilder(); $container->register('a', __NAMESPACE__.'\A'); $container->register('lille', __NAMESPACE__.'\Lille'); $container->register('with_optional_scalar', __NAMESPACE__.'\MultipleArgumentsOptionalScalar') ->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $definition = $container->getDefinition('with_optional_scalar'); $this->assertEquals( array( new Reference('a'), // use the default value 'default_val', new Reference('lille'), ), $definition->getArguments() ); } public function testOptionalScalarArgsNotPassedIfLast() { $container = new ContainerBuilder(); $container->register('a', __NAMESPACE__.'\A'); $container->register('lille', __NAMESPACE__.'\Lille'); $container->register('with_optional_scalar_last', __NAMESPACE__.'\MultipleArgumentsOptionalScalarLast') ->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $definition = $container->getDefinition('with_optional_scalar_last'); $this->assertEquals( array( new Reference('a'), new Reference('lille'), ), $definition->getArguments() ); } public function testIgnoreServiceWithClassNotExisting() { $container = new ContainerBuilder(); $container->register('class_not_exist', __NAMESPACE__.'\OptionalServiceClass'); $barDefinition = $container->register('bar', __NAMESPACE__.'\Bar'); $barDefinition->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $this->assertTrue($container->hasDefinition('bar')); } public function testProcessDoesNotTriggerDeprecations() { $container = new ContainerBuilder(); $container->register('deprecated', 'Symfony\Component\DependencyInjection\Tests\Fixtures\DeprecatedClass')->setDeprecated(true); $container->register('foo', __NAMESPACE__.'\Foo'); $container->register('bar', __NAMESPACE__.'\Bar')->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); $this->assertTrue($container->hasDefinition('deprecated')); $this->assertTrue($container->hasDefinition('foo')); $this->assertTrue($container->hasDefinition('bar')); } public function testEmptyStringIsKept() { $container = new ContainerBuilder(); $container->register('a', __NAMESPACE__.'\A'); $container->register('lille', __NAMESPACE__.'\Lille'); $container->register('foo', __NAMESPACE__.'\MultipleArgumentsOptionalScalar') ->setAutowired(true) ->setArguments(array('', '')); $pass = new AutowirePass(); $pass->process($container); $this->assertEquals(array(new Reference('a'), '', new Reference('lille')), $container->getDefinition('foo')->getArguments()); } public function provideAutodiscoveredAutowiringOrder() { return array( array('CannotBeAutowiredForwardOrder'), array('CannotBeAutowiredReverseOrder'), ); } /** * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException * @expectedExceptionMessage Service "a" can use either autowiring or a factory, not both. */ public function testWithFactory() { $container = new ContainerBuilder(); $container->register('a', __NAMESPACE__.'\A') ->setFactory('foo') ->setAutowired(true); $pass = new AutowirePass(); $pass->process($container); } } class Foo { } class Bar { public function __construct(Foo $foo) { } } class A { } class B extends A { } class C { public function __construct(A $a) { } } interface DInterface { } interface EInterface extends DInterface { } interface IInterface { } class I implements IInterface { } class F extends I implements EInterface { } class G { public function __construct(DInterface $d, EInterface $e, IInterface $i) { } } class H { public function __construct(B $b, DInterface $d) { } } interface CollisionInterface { } class CollisionA implements CollisionInterface { } class CollisionB implements CollisionInterface { } class CannotBeAutowired { public function __construct(CollisionInterface $collision) { } } class CannotBeAutowiredForwardOrder { public function __construct(CollisionA $a, CollisionInterface $b, CollisionB $c) { } } class CannotBeAutowiredReverseOrder { public function __construct(CollisionA $a, CollisionB $c, CollisionInterface $b) { } } class Lille { } class Dunglas { public function __construct(Lille $l) { } } class LesTilleuls { public function __construct(Dunglas $j, Dunglas $k) { } } class OptionalParameter { public function __construct(CollisionInterface $c = null, A $a, Foo $f = null) { } } class BadTypeHintedArgument { public function __construct(Dunglas $k, NotARealClass $r) { } } class BadParentTypeHintedArgument { public function __construct(Dunglas $k, OptionalServiceClass $r) { } } class NotGuessableArgument { public function __construct(Foo $k) { } } class NotGuessableArgumentForSubclass { public function __construct(A $k) { } } class MultipleArguments { public function __construct(A $k, $foo, Dunglas $dunglas) { } } class MultipleArgumentsOptionalScalar { public function __construct(A $a, $foo = 'default_val', Lille $lille = null) { } } class MultipleArgumentsOptionalScalarLast { public function __construct(A $a, Lille $lille, $foo = 'some_val') { } } class MultipleArgumentsOptionalScalarNotReallyOptional { public function __construct(A $a, $foo = 'default_val', Lille $lille) { } }