require_once __DIR__.'/Fixtures/includes/ProjectExtension.php';
use PHPUnit\Framework\TestCase;
+use Psr\Container\ContainerInterface as PsrContainerInterface;
+use Symfony\Component\Config\Resource\ComposerResource;
use Symfony\Component\Config\Resource\ResourceInterface;
+use Symfony\Component\Config\Resource\DirectoryResource;
use Symfony\Component\DependencyInjection\Alias;
+use Symfony\Component\DependencyInjection\Argument\IteratorArgument;
+use Symfony\Component\DependencyInjection\Argument\RewindableGenerator;
+use Symfony\Component\DependencyInjection\Argument\ServiceClosureArgument;
+use Symfony\Component\DependencyInjection\ChildDefinition;
use Symfony\Component\DependencyInjection\Compiler\PassConfig;
use Symfony\Component\DependencyInjection\ContainerBuilder;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\DependencyInjection\Definition;
-use Symfony\Component\DependencyInjection\DefinitionDecorator;
use Symfony\Component\DependencyInjection\Exception\RuntimeException;
use Symfony\Component\DependencyInjection\Exception\ServiceNotFoundException;
use Symfony\Component\DependencyInjection\Loader\ClosureLoader;
use Symfony\Component\DependencyInjection\Reference;
+use Symfony\Component\DependencyInjection\Tests\Fixtures\SimilarArgumentsDummy;
+use Symfony\Component\DependencyInjection\TypedReference;
use Symfony\Component\DependencyInjection\ParameterBag\ParameterBag;
use Symfony\Component\DependencyInjection\ParameterBag\EnvPlaceholderParameterBag;
use Symfony\Component\Config\Resource\FileResource;
+use Symfony\Component\DependencyInjection\ServiceLocator;
use Symfony\Component\DependencyInjection\Tests\Fixtures\CustomDefinition;
+use Symfony\Component\DependencyInjection\Tests\Fixtures\CaseSensitiveClass;
use Symfony\Component\ExpressionLanguage\Expression;
class ContainerBuilderTest extends TestCase
{
+ public function testDefaultRegisteredDefinitions()
+ {
+ $builder = new ContainerBuilder();
+
+ $this->assertCount(1, $builder->getDefinitions());
+ $this->assertTrue($builder->hasDefinition('service_container'));
+
+ $definition = $builder->getDefinition('service_container');
+ $this->assertInstanceOf(Definition::class, $definition);
+ $this->assertTrue($definition->isSynthetic());
+ $this->assertSame(ContainerInterface::class, $definition->getClass());
+ $this->assertTrue($builder->hasAlias(PsrContainerInterface::class));
+ $this->assertTrue($builder->hasAlias(ContainerInterface::class));
+ }
+
public function testDefinitions()
{
$builder = new ContainerBuilder();
$builder->setDefinition('foobar', $foo = new Definition('FooBarClass'));
$this->assertEquals($foo, $builder->getDefinition('foobar'), '->getDefinition() returns a service definition if defined');
- $this->assertTrue($builder->setDefinition('foobar', $foo = new Definition('FooBarClass')) === $foo, '->setDefinition() implements a fluid interface by returning the service reference');
+ $this->assertSame($builder->setDefinition('foobar', $foo = new Definition('FooBarClass')), $foo, '->setDefinition() implements a fluid interface by returning the service reference');
$builder->addDefinitions($defs = array('foobar' => new Definition('FooBarClass')));
$this->assertEquals(array_merge($definitions, $defs), $builder->getDefinitions(), '->addDefinitions() adds the service definitions');
$this->assertInstanceOf('Symfony\Component\DependencyInjection\Definition', $builder->getDefinition('foo'), '->register() returns the newly created Definition instance');
}
+ public function testAutowire()
+ {
+ $builder = new ContainerBuilder();
+ $builder->autowire('foo', 'Bar\FooClass');
+
+ $this->assertTrue($builder->hasDefinition('foo'), '->autowire() registers a new service definition');
+ $this->assertTrue($builder->getDefinition('foo')->isAutowired(), '->autowire() creates autowired definitions');
+ }
+
public function testHas()
{
$builder = new ContainerBuilder();
$builder->register('foo', 'stdClass');
$builder->bar = $bar = new \stdClass();
$builder->register('bar', 'stdClass');
- $this->assertEquals(array('foo', 'bar', 'service_container'), $builder->getServiceIds(), '->getServiceIds() returns all defined service ids');
+ $this->assertEquals(
+ array(
+ 'service_container',
+ 'foo',
+ 'bar',
+ 'Psr\Container\ContainerInterface',
+ 'Symfony\Component\DependencyInjection\ContainerInterface',
+ ),
+ $builder->getServiceIds(),
+ '->getServiceIds() returns all defined service ids'
+ );
}
public function testAliases()
$this->assertFalse($builder->hasAlias('foobar'), '->hasAlias() returns false if the alias does not exist');
$this->assertEquals('foo', (string) $builder->getAlias('bar'), '->getAlias() returns the aliased service');
$this->assertTrue($builder->has('bar'), '->setAlias() defines a new service');
- $this->assertTrue($builder->get('bar') === $builder->get('foo'), '->setAlias() creates a service that is an alias to another one');
+ $this->assertSame($builder->get('bar'), $builder->get('foo'), '->setAlias() creates a service that is an alias to another one');
try {
$builder->setAlias('foobar', 'foobar');
$builder->set('foobar', 'stdClass');
$builder->set('moo', 'stdClass');
- $this->assertCount(0, $builder->getAliases(), '->getAliases() does not return aliased services that have been overridden');
+ $this->assertCount(2, $builder->getAliases(), '->getAliases() does not return aliased services that have been overridden');
}
public function testSetAliases()
$builder->setAliases(array('bar' => 'foo', 'foobar' => 'foo'));
$aliases = $builder->getAliases();
- $this->assertTrue(isset($aliases['bar']));
- $this->assertTrue(isset($aliases['foobar']));
+ $this->assertArrayHasKey('bar', $aliases);
+ $this->assertArrayHasKey('foobar', $aliases);
}
public function testAddAliases()
$builder->addAliases(array('foobar' => 'foo'));
$aliases = $builder->getAliases();
- $this->assertTrue(isset($aliases['bar']));
- $this->assertTrue(isset($aliases['foobar']));
+ $this->assertArrayHasKey('bar', $aliases);
+ $this->assertArrayHasKey('foobar', $aliases);
}
public function testSetReplacesAlias()
}
}
+ public function testCreateServiceWithIteratorArgument()
+ {
+ $builder = new ContainerBuilder();
+ $builder->register('bar', 'stdClass');
+ $builder
+ ->register('lazy_context', 'LazyContext')
+ ->setArguments(array(
+ new IteratorArgument(array('k1' => new Reference('bar'), new Reference('invalid', ContainerInterface::IGNORE_ON_INVALID_REFERENCE))),
+ new IteratorArgument(array()),
+ ))
+ ;
+
+ $lazyContext = $builder->get('lazy_context');
+ $this->assertInstanceOf(RewindableGenerator::class, $lazyContext->lazyValues);
+ $this->assertInstanceOf(RewindableGenerator::class, $lazyContext->lazyEmptyValues);
+ $this->assertCount(1, $lazyContext->lazyValues);
+ $this->assertCount(0, $lazyContext->lazyEmptyValues);
+
+ $i = 0;
+ foreach ($lazyContext->lazyValues as $k => $v) {
+ ++$i;
+ $this->assertEquals('k1', $k);
+ $this->assertInstanceOf('\stdClass', $v);
+ }
+
+ // The second argument should have been ignored.
+ $this->assertEquals(1, $i);
+
+ $i = 0;
+ foreach ($lazyContext->lazyEmptyValues as $k => $v) {
+ ++$i;
+ }
+
+ $this->assertEquals(0, $i);
+ }
+
/**
* @expectedException \RuntimeException
*/
{
$builder = new ContainerBuilder();
$builder->setDefinition('grandpa', new Definition('stdClass'));
- $builder->setDefinition('parent', new DefinitionDecorator('grandpa'));
- $builder->setDefinition('foo', new DefinitionDecorator('parent'));
+ $builder->setDefinition('parent', new ChildDefinition('grandpa'));
+ $builder->setDefinition('foo', new ChildDefinition('parent'));
$builder->get('foo');
}
$config->setDefinition('baz', new Definition('BazClass'));
$config->setAlias('alias_for_foo', 'foo');
$container->merge($config);
- $this->assertEquals(array('foo', 'bar', 'baz'), array_keys($container->getDefinitions()), '->merge() merges definitions already defined ones');
+ $this->assertEquals(array('service_container', 'foo', 'bar', 'baz'), array_keys($container->getDefinitions()), '->merge() merges definitions already defined ones');
$aliases = $container->getAliases();
- $this->assertTrue(isset($aliases['alias_for_foo']));
+ $this->assertArrayHasKey('alias_for_foo', $aliases);
$this->assertEquals('foo', (string) $aliases['alias_for_foo']);
$container = new ContainerBuilder();
$this->assertSame(array('%env(Bar)%'), $config->resolveEnvPlaceholders(array($bag->get('env(Bar)'))));
$container->merge($config);
$this->assertEquals(array('Foo' => 0, 'Bar' => 1), $container->getEnvCounters());
+
+ $container = new ContainerBuilder();
+ $config = new ContainerBuilder();
+ $childDefA = $container->registerForAutoconfiguration('AInterface');
+ $childDefB = $config->registerForAutoconfiguration('BInterface');
+ $container->merge($config);
+ $this->assertSame(array('AInterface' => $childDefA, 'BInterface' => $childDefB), $container->getAutoconfiguredInstanceof());
+ }
+
+ /**
+ * @expectedException \Symfony\Component\DependencyInjection\Exception\InvalidArgumentException
+ * @expectedExceptionMessage "AInterface" has already been autoconfigured and merge() does not support merging autoconfiguration for the same class/interface.
+ */
+ public function testMergeThrowsExceptionForDuplicateAutomaticInstanceofDefinitions()
+ {
+ $container = new ContainerBuilder();
+ $config = new ContainerBuilder();
+ $container->registerForAutoconfiguration('AInterface');
+ $config->registerForAutoconfiguration('AInterface');
+ $container->merge($config);
+ }
+
+ public function testResolveEnvValues()
+ {
+ $_ENV['DUMMY_ENV_VAR'] = 'du%%y';
+ $_SERVER['DUMMY_SERVER_VAR'] = 'ABC';
+ $_SERVER['HTTP_DUMMY_VAR'] = 'DEF';
+
+ $container = new ContainerBuilder();
+ $container->setParameter('bar', '%% %env(DUMMY_ENV_VAR)% %env(DUMMY_SERVER_VAR)% %env(HTTP_DUMMY_VAR)%');
+ $container->setParameter('env(HTTP_DUMMY_VAR)', '123');
+
+ $this->assertSame('%% du%%%%y ABC 123', $container->resolveEnvPlaceholders('%bar%', true));
+
+ unset($_ENV['DUMMY_ENV_VAR'], $_SERVER['DUMMY_SERVER_VAR'], $_SERVER['HTTP_DUMMY_VAR']);
+ }
+
+ public function testResolveEnvValuesWithArray()
+ {
+ $_ENV['ANOTHER_DUMMY_ENV_VAR'] = 'dummy';
+
+ $dummyArray = array('1' => 'one', '2' => 'two');
+
+ $container = new ContainerBuilder();
+ $container->setParameter('dummy', '%env(ANOTHER_DUMMY_ENV_VAR)%');
+ $container->setParameter('dummy2', $dummyArray);
+
+ $container->resolveEnvPlaceholders('%dummy%', true);
+ $container->resolveEnvPlaceholders('%dummy2%', true);
+
+ $this->assertInternalType('array', $container->resolveEnvPlaceholders('%dummy2%', true));
+
+ foreach ($dummyArray as $key => $value) {
+ $this->assertArrayHasKey($key, $container->resolveEnvPlaceholders('%dummy2%', true));
+ }
+
+ unset($_ENV['ANOTHER_DUMMY_ENV_VAR']);
+ }
+
+ public function testCompileWithResolveEnv()
+ {
+ putenv('DUMMY_ENV_VAR=du%%y');
+ $_SERVER['DUMMY_SERVER_VAR'] = 'ABC';
+ $_SERVER['HTTP_DUMMY_VAR'] = 'DEF';
+
+ $container = new ContainerBuilder();
+ $container->setParameter('env(FOO)', 'Foo');
+ $container->setParameter('env(DUMMY_ENV_VAR)', 'GHI');
+ $container->setParameter('bar', '%% %env(DUMMY_ENV_VAR)% %env(DUMMY_SERVER_VAR)% %env(HTTP_DUMMY_VAR)%');
+ $container->setParameter('foo', '%env(FOO)%');
+ $container->setParameter('baz', '%foo%');
+ $container->setParameter('env(HTTP_DUMMY_VAR)', '123');
+ $container->register('teatime', 'stdClass')
+ ->setProperty('foo', '%env(DUMMY_ENV_VAR)%')
+ ->setPublic(true)
+ ;
+ $container->compile(true);
+
+ $this->assertSame('% du%%y ABC 123', $container->getParameter('bar'));
+ $this->assertSame('Foo', $container->getParameter('baz'));
+ $this->assertSame('du%%y', $container->get('teatime')->foo);
+
+ unset($_SERVER['DUMMY_SERVER_VAR'], $_SERVER['HTTP_DUMMY_VAR']);
+ putenv('DUMMY_ENV_VAR');
+ }
+
+ /**
+ * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException
+ * @expectedExceptionMessage A string value must be composed of strings and/or numbers, but found parameter "env(ARRAY)" of type array inside string value "ABC %env(ARRAY)%".
+ */
+ public function testCompileWithArrayResolveEnv()
+ {
+ $bag = new TestingEnvPlaceholderParameterBag();
+ $container = new ContainerBuilder($bag);
+ $container->setParameter('foo', '%env(ARRAY)%');
+ $container->setParameter('bar', 'ABC %env(ARRAY)%');
+ $container->compile(true);
+ }
+
+ /**
+ * @expectedException \Symfony\Component\DependencyInjection\Exception\EnvNotFoundException
+ * @expectedExceptionMessage Environment variable not found: "FOO".
+ */
+ public function testCompileWithResolveMissingEnv()
+ {
+ $container = new ContainerBuilder();
+ $container->setParameter('foo', '%env(FOO)%');
+ $container->compile(true);
+ }
+
+ public function testDynamicEnv()
+ {
+ putenv('DUMMY_FOO=some%foo%');
+ putenv('DUMMY_BAR=%bar%');
+
+ $container = new ContainerBuilder();
+ $container->setParameter('foo', 'Foo%env(resolve:DUMMY_BAR)%');
+ $container->setParameter('bar', 'Bar');
+ $container->setParameter('baz', '%env(resolve:DUMMY_FOO)%');
+
+ $container->compile(true);
+ putenv('DUMMY_FOO');
+ putenv('DUMMY_BAR');
+
+ $this->assertSame('someFooBar', $container->getParameter('baz'));
+ }
+
+ public function testCastEnv()
+ {
+ $container = new ContainerBuilder();
+ $container->setParameter('env(FAKE)', '123');
+
+ $container->register('foo', 'stdClass')
+ ->setPublic(true)
+ ->setProperties(array(
+ 'fake' => '%env(int:FAKE)%',
+ ));
+
+ $container->compile(true);
+
+ $this->assertSame(123, $container->get('foo')->fake);
+ }
+
+ public function testEnvAreNullable()
+ {
+ $container = new ContainerBuilder();
+ $container->setParameter('env(FAKE)', null);
+
+ $container->register('foo', 'stdClass')
+ ->setPublic(true)
+ ->setProperties(array(
+ 'fake' => '%env(int:FAKE)%',
+ ));
+
+ $container->compile(true);
+
+ $this->assertNull($container->get('foo')->fake);
+ }
+
+ public function testEnvInId()
+ {
+ $container = include __DIR__.'/Fixtures/containers/container_env_in_id.php';
+ $container->compile(true);
+
+ $expected = array(
+ 'service_container',
+ 'foo',
+ 'bar',
+ 'bar_%env(BAR)%',
+ );
+ $this->assertSame($expected, array_keys($container->getDefinitions()));
+
+ $expected = array(
+ PsrContainerInterface::class => true,
+ ContainerInterface::class => true,
+ 'baz_%env(BAR)%' => true,
+ 'bar_%env(BAR)%' => true,
+ );
+ $this->assertSame($expected, $container->getRemovedIds());
+
+ $this->assertSame(array('baz_bar'), array_keys($container->getDefinition('foo')->getArgument(1)));
+ }
+
+ /**
+ * @expectedException \Symfony\Component\DependencyInjection\Exception\ParameterCircularReferenceException
+ * @expectedExceptionMessage Circular reference detected for parameter "env(resolve:DUMMY_ENV_VAR)" ("env(resolve:DUMMY_ENV_VAR)" > "env(resolve:DUMMY_ENV_VAR)").
+ */
+ public function testCircularDynamicEnv()
+ {
+ putenv('DUMMY_ENV_VAR=some%foo%');
+
+ $container = new ContainerBuilder();
+ $container->setParameter('foo', '%bar%');
+ $container->setParameter('bar', '%env(resolve:DUMMY_ENV_VAR)%');
+
+ try {
+ $container->compile(true);
+ } finally {
+ putenv('DUMMY_ENV_VAR');
+ }
}
/**
$resources = $container->getResources();
- $this->assertCount(1, $resources, '1 resource was registered');
+ $this->assertCount(2, $resources, '2 resources were registered');
/* @var $resource \Symfony\Component\Config\Resource\FileResource */
$resource = end($resources);
$this->assertSame(realpath(__DIR__.'/Fixtures/includes/classes.php'), realpath($resource->getResource()));
}
+ /**
+ * @group legacy
+ */
public function testAddClassResource()
{
$container = new ContainerBuilder();
$resources = $container->getResources();
- $this->assertCount(1, $resources, '1 resource was registered');
+ $this->assertCount(2, $resources, '2 resources were registered');
/* @var $resource \Symfony\Component\Config\Resource\FileResource */
$resource = end($resources);
$this->assertSame(realpath(__DIR__.'/Fixtures/includes/classes.php'), realpath($resource->getResource()));
}
+ public function testGetReflectionClass()
+ {
+ $container = new ContainerBuilder();
+
+ $container->setResourceTracking(false);
+ $r1 = $container->getReflectionClass('BarClass');
+
+ $this->assertEmpty($container->getResources(), 'No resources get registered without resource tracking');
+
+ $container->setResourceTracking(true);
+ $r2 = $container->getReflectionClass('BarClass');
+ $r3 = $container->getReflectionClass('BarClass');
+
+ $this->assertNull($container->getReflectionClass('BarMissingClass'));
+
+ $this->assertEquals($r1, $r2);
+ $this->assertSame($r2, $r3);
+
+ $resources = $container->getResources();
+
+ $this->assertCount(3, $resources, '3 resources were registered');
+
+ $this->assertSame('reflection.BarClass', (string) $resources[1]);
+ $this->assertSame('BarMissingClass', (string) end($resources));
+ }
+
+ public function testGetReflectionClassOnInternalTypes()
+ {
+ $container = new ContainerBuilder();
+
+ $this->assertNull($container->getReflectionClass('int'));
+ $this->assertNull($container->getReflectionClass('float'));
+ $this->assertNull($container->getReflectionClass('string'));
+ $this->assertNull($container->getReflectionClass('bool'));
+ $this->assertNull($container->getReflectionClass('resource'));
+ $this->assertNull($container->getReflectionClass('object'));
+ $this->assertNull($container->getReflectionClass('array'));
+ $this->assertNull($container->getReflectionClass('null'));
+ $this->assertNull($container->getReflectionClass('callable'));
+ $this->assertNull($container->getReflectionClass('iterable'));
+ $this->assertNull($container->getReflectionClass('mixed'));
+ }
+
public function testCompilesClassDefinitionsOfLazyServices()
{
$container = new ContainerBuilder();
$this->assertEmpty($container->getResources(), 'No resources get registered without resource tracking');
- $container->register('foo', 'BarClass');
+ $container->register('foo', 'BarClass')->setPublic(true);
$container->getDefinition('foo')->setLazy(true);
$container->compile();
- $classesPath = realpath(__DIR__.'/Fixtures/includes/classes.php');
$matchingResources = array_filter(
$container->getResources(),
- function (ResourceInterface $resource) use ($classesPath) {
- return $resource instanceof FileResource && $classesPath === realpath($resource->getResource());
+ function (ResourceInterface $resource) {
+ return 'reflection.BarClass' === (string) $resource;
}
);
$this->assertEquals(array(), $container->getResources());
}
+ public function testFileExists()
+ {
+ $container = new ContainerBuilder();
+ $A = new ComposerResource();
+ $a = new FileResource(__DIR__.'/Fixtures/xml/services1.xml');
+ $b = new FileResource(__DIR__.'/Fixtures/xml/services2.xml');
+ $c = new DirectoryResource($dir = dirname($b));
+
+ $this->assertTrue($container->fileExists((string) $a) && $container->fileExists((string) $b) && $container->fileExists($dir));
+
+ $resources = array();
+ foreach ($container->getResources() as $resource) {
+ if (false === strpos($resource, '.php')) {
+ $resources[] = $resource;
+ }
+ }
+
+ $this->assertEquals(array($A, $a, $b, $c), $resources, '->getResources() returns an array of resources read for the current configuration');
+ }
+
public function testExtension()
{
$container = new ContainerBuilder();
$container->setResourceTracking(false);
$container->registerExtension($extension = new \ProjectExtension());
- $this->assertTrue($container->getExtension('project') === $extension, '->registerExtension() registers an extension');
+ $this->assertSame($container->getExtension('project'), $extension, '->registerExtension() registers an extension');
$this->{method_exists($this, $_ = 'expectException') ? $_ : 'setExpectedException'}('LogicException');
$container->getExtension('no_registered');
$container->addDefinitions(array(
'bar' => $fooDefinition,
- 'bar_user' => $fooUserDefinition,
+ 'bar_user' => $fooUserDefinition->setPublic(true),
));
$container->compile();
/**
* @expectedException \BadMethodCallException
*/
- public function testThrowsExceptionWhenSetServiceOnAFrozenContainer()
+ public function testThrowsExceptionWhenSetServiceOnACompiledContainer()
{
$container = new ContainerBuilder();
$container->setResourceTracking(false);
- $container->setDefinition('a', new Definition('stdClass'));
+ $container->register('a', 'stdClass')->setPublic(true);
$container->compile();
$container->set('a', new \stdClass());
}
- public function testThrowsExceptionWhenAddServiceOnAFrozenContainer()
+ public function testThrowsExceptionWhenAddServiceOnACompiledContainer()
{
$container = new ContainerBuilder();
$container->compile();
$this->assertSame($foo, $container->get('a'));
}
- public function testNoExceptionWhenSetSyntheticServiceOnAFrozenContainer()
+ public function testNoExceptionWhenSetSyntheticServiceOnACompiledContainer()
{
$container = new ContainerBuilder();
$def = new Definition('stdClass');
- $def->setSynthetic(true);
+ $def->setSynthetic(true)->setPublic(true);
$container->setDefinition('a', $def);
$container->compile();
$container->set('a', $a = new \stdClass());
/**
* @expectedException \BadMethodCallException
*/
- public function testThrowsExceptionWhenSetDefinitionOnAFrozenContainer()
+ public function testThrowsExceptionWhenSetDefinitionOnACompiledContainer()
{
$container = new ContainerBuilder();
$container->setResourceTracking(false);
$container = new ContainerBuilder();
$abstract = new Definition('AbstractClass');
- $abstract->setAbstract(true);
+ $abstract->setAbstract(true)->setPublic(true);
$container->setDefinition('abstract_service', $abstract);
- $container->setAlias('abstract_alias', 'abstract_service');
+ $container->setAlias('abstract_alias', 'abstract_service')->setPublic(true);
$container->compile();
$container->set('a', new \BazClass());
$definition = new Definition('BazClass');
$definition->setLazy(true);
+ $definition->setPublic(true);
$container->setDefinition('a', $definition);
});
$container->compile();
- $class = new \BazClass();
- $reflectionClass = new \ReflectionClass($class);
-
$r = new \ReflectionProperty($container, 'resources');
$r->setAccessible(true);
$resources = $r->getValue($container);
$classInList = false;
foreach ($resources as $resource) {
- if ($resource->getResource() === $reflectionClass->getFileName()) {
+ if ('reflection.BazClass' === (string) $resource) {
$classInList = true;
break;
}
$this->assertTrue($classInList);
}
+ public function testInlinedDefinitions()
+ {
+ $container = new ContainerBuilder();
+
+ $definition = new Definition('BarClass');
+
+ $container->register('bar_user', 'BarUserClass')
+ ->addArgument($definition)
+ ->setProperty('foo', $definition);
+
+ $container->register('bar', 'BarClass')
+ ->setProperty('foo', $definition)
+ ->addMethodCall('setBaz', array($definition));
+
+ $barUser = $container->get('bar_user');
+ $bar = $container->get('bar');
+
+ $this->assertSame($barUser->foo, $barUser->bar);
+ $this->assertSame($bar->foo, $bar->getBaz());
+ $this->assertNotSame($bar->foo, $barUser->foo);
+ }
+
+ /**
+ * @expectedException \Symfony\Component\DependencyInjection\Exception\ServiceCircularReferenceException
+ * @expectedExceptionMessage Circular reference detected for service "app.test_class", path: "app.test_class -> App\TestClass -> app.test_class".
+ */
+ public function testThrowsCircularExceptionForCircularAliases()
+ {
+ $builder = new ContainerBuilder();
+
+ $builder->setAliases(array(
+ 'foo' => new Alias('app.test_class'),
+ 'app.test_class' => new Alias('App\\TestClass'),
+ 'App\\TestClass' => new Alias('app.test_class'),
+ ));
+
+ $builder->findDefinition('foo');
+ }
+
public function testInitializePropertiesBeforeMethodCalls()
{
$container = new ContainerBuilder();
$container->register('foo', 'stdClass');
$container->register('bar', 'MethodCallClass')
+ ->setPublic(true)
->setProperty('simple', 'bar')
->setProperty('complex', new Reference('foo'))
->addMethodCall('callMe');
{
$container = new ContainerBuilder();
- $container->register('a', __NAMESPACE__.'\A');
+ $container->register(A::class)->setPublic(true);
$bDefinition = $container->register('b', __NAMESPACE__.'\B');
$bDefinition->setAutowired(true);
+ $bDefinition->setPublic(true);
+
+ $container->compile();
+
+ $this->assertEquals(A::class, (string) $container->getDefinition('b')->getArgument(0));
+ }
+
+ public function testClassFromId()
+ {
+ $container = new ContainerBuilder();
+
+ $unknown = $container->register('Acme\UnknownClass');
+ $autoloadClass = $container->register(CaseSensitiveClass::class);
+ $container->compile();
+
+ $this->assertSame('Acme\UnknownClass', $unknown->getClass());
+ $this->assertEquals(CaseSensitiveClass::class, $autoloadClass->getClass());
+ }
+
+ /**
+ * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException
+ * @expectedExceptionMessage The definition for "DateTime" has no class attribute, and appears to reference a class or interface in the global namespace.
+ */
+ public function testNoClassFromGlobalNamespaceClassId()
+ {
+ $container = new ContainerBuilder();
+ $definition = $container->register(\DateTime::class);
$container->compile();
+ }
- $this->assertEquals('a', (string) $container->getDefinition('b')->getArgument(0));
+ /**
+ * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException
+ * @expectedExceptionMessage The definition for "123_abc" has no class.
+ */
+ public function testNoClassFromNonClassId()
+ {
+ $container = new ContainerBuilder();
+
+ $definition = $container->register('123_abc');
+ $container->compile();
+ }
+
+ /**
+ * @expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException
+ * @expectedExceptionMessage The definition for "\foo" has no class.
+ */
+ public function testNoClassFromNsSeparatorId()
+ {
+ $container = new ContainerBuilder();
+
+ $definition = $container->register('\\foo');
+ $container->compile();
+ }
+
+ public function testServiceLocator()
+ {
+ $container = new ContainerBuilder();
+ $container->register('foo_service', ServiceLocator::class)
+ ->setPublic(true)
+ ->addArgument(array(
+ 'bar' => new ServiceClosureArgument(new Reference('bar_service')),
+ 'baz' => new ServiceClosureArgument(new TypedReference('baz_service', 'stdClass')),
+ ))
+ ;
+ $container->register('bar_service', 'stdClass')->setArguments(array(new Reference('baz_service')))->setPublic(true);
+ $container->register('baz_service', 'stdClass')->setPublic(false);
+ $container->compile();
+
+ $this->assertInstanceOf(ServiceLocator::class, $foo = $container->get('foo_service'));
+ $this->assertSame($container->get('bar_service'), $foo->get('bar'));
+ }
+
+ public function testUninitializedReference()
+ {
+ $container = include __DIR__.'/Fixtures/containers/container_uninitialized_ref.php';
+ $container->compile();
+
+ $bar = $container->get('bar');
+
+ $this->assertNull($bar->foo1);
+ $this->assertNull($bar->foo2);
+ $this->assertNull($bar->foo3);
+ $this->assertNull($bar->closures[0]());
+ $this->assertNull($bar->closures[1]());
+ $this->assertNull($bar->closures[2]());
+ $this->assertSame(array(), iterator_to_array($bar->iter));
+
+ $container = include __DIR__.'/Fixtures/containers/container_uninitialized_ref.php';
+ $container->compile();
+
+ $container->get('foo1');
+ $container->get('baz');
+
+ $bar = $container->get('bar');
+
+ $this->assertEquals(new \stdClass(), $bar->foo1);
+ $this->assertNull($bar->foo2);
+ $this->assertEquals(new \stdClass(), $bar->foo3);
+ $this->assertEquals(new \stdClass(), $bar->closures[0]());
+ $this->assertNull($bar->closures[1]());
+ $this->assertEquals(new \stdClass(), $bar->closures[2]());
+ $this->assertEquals(array('foo1' => new \stdClass(), 'foo3' => new \stdClass()), iterator_to_array($bar->iter));
+ }
+
+ /**
+ * @dataProvider provideAlmostCircular
+ */
+ public function testAlmostCircular($visibility)
+ {
+ $container = include __DIR__.'/Fixtures/containers/container_almost_circular.php';
+
+ $foo = $container->get('foo');
+ $this->assertSame($foo, $foo->bar->foobar->foo);
+
+ $foo2 = $container->get('foo2');
+ $this->assertSame($foo2, $foo2->bar->foobar->foo);
+
+ $this->assertSame(array(), (array) $container->get('foobar4'));
+
+ $foo5 = $container->get('foo5');
+ $this->assertSame($foo5, $foo5->bar->foo);
+ }
+
+ public function provideAlmostCircular()
+ {
+ yield array('public');
+ yield array('private');
+ }
+
+ public function testRegisterForAutoconfiguration()
+ {
+ $container = new ContainerBuilder();
+ $childDefA = $container->registerForAutoconfiguration('AInterface');
+ $childDefB = $container->registerForAutoconfiguration('BInterface');
+ $this->assertSame(array('AInterface' => $childDefA, 'BInterface' => $childDefB), $container->getAutoconfiguredInstanceof());
+
+ // when called multiple times, the same instance is returned
+ $this->assertSame($childDefA, $container->registerForAutoconfiguration('AInterface'));
}
/**
$container->get('bar');
}
+
+ /**
+ * @group legacy
+ * @expectedDeprecation Parameter names will be made case sensitive in Symfony 4.0. Using "FOO" instead of "foo" is deprecated since Symfony 3.4.
+ */
+ public function testParameterWithMixedCase()
+ {
+ $container = new ContainerBuilder(new ParameterBag(array('foo' => 'bar')));
+ $container->register('foo', 'stdClass')
+ ->setPublic(true)
+ ->setProperty('foo', '%FOO%');
+
+ $container->compile();
+
+ $this->assertSame('bar', $container->get('foo')->foo);
+ }
+
+ public function testArgumentsHaveHigherPriorityThanBindings()
+ {
+ $container = new ContainerBuilder();
+ $container->register('class.via.bindings', CaseSensitiveClass::class)->setArguments(array(
+ 'via-bindings',
+ ));
+ $container->register('class.via.argument', CaseSensitiveClass::class)->setArguments(array(
+ 'via-argument',
+ ));
+ $container->register('foo', SimilarArgumentsDummy::class)->setPublic(true)->setBindings(array(
+ CaseSensitiveClass::class => new Reference('class.via.bindings'),
+ '$token' => '1234',
+ ))->setArguments(array(
+ '$class1' => new Reference('class.via.argument'),
+ ));
+
+ $this->assertSame(array('service_container', 'class.via.bindings', 'class.via.argument', 'foo', 'Psr\Container\ContainerInterface', 'Symfony\Component\DependencyInjection\ContainerInterface'), $container->getServiceIds());
+
+ $container->compile();
+
+ $this->assertSame('via-argument', $container->get('foo')->class1->identifier);
+ $this->assertSame('via-bindings', $container->get('foo')->class2->identifier);
+ }
}
class FooClass
{
}
}
+
+class TestingEnvPlaceholderParameterBag extends EnvPlaceholderParameterBag
+{
+ public function get($name)
+ {
+ return 'env(array)' === strtolower($name) ? array(123) : parent::get($name);
+ }
+}