2018-11-23 12:29:20 +00:00
< ? php
/*
* This file is part of the Symfony package .
*
* ( c ) Fabien Potencier < fabien @ symfony . com >
*
* For the full copyright and license information , please view the LICENSE
* file that was distributed with this source code .
*/
namespace Symfony\Component\DependencyInjection\Tests ;
require_once __DIR__ . '/Fixtures/includes/classes.php' ;
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\DirectoryResource ;
use Symfony\Component\Config\Resource\FileResource ;
use Symfony\Component\Config\Resource\ResourceInterface ;
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\Exception\RuntimeException ;
use Symfony\Component\DependencyInjection\Exception\ServiceNotFoundException ;
use Symfony\Component\DependencyInjection\Loader\ClosureLoader ;
use Symfony\Component\DependencyInjection\ParameterBag\EnvPlaceholderParameterBag ;
use Symfony\Component\DependencyInjection\ParameterBag\ParameterBag ;
use Symfony\Component\DependencyInjection\Reference ;
use Symfony\Component\DependencyInjection\ServiceLocator ;
use Symfony\Component\DependencyInjection\Tests\Fixtures\CaseSensitiveClass ;
use Symfony\Component\DependencyInjection\Tests\Fixtures\CustomDefinition ;
use Symfony\Component\DependencyInjection\Tests\Fixtures\SimilarArgumentsDummy ;
use Symfony\Component\DependencyInjection\TypedReference ;
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 ();
$definitions = array (
'foo' => new Definition ( 'Bar\FooClass' ),
'bar' => new Definition ( 'BarClass' ),
);
$builder -> setDefinitions ( $definitions );
$this -> assertEquals ( $definitions , $builder -> getDefinitions (), '->setDefinitions() sets the service definitions' );
$this -> assertTrue ( $builder -> hasDefinition ( 'foo' ), '->hasDefinition() returns true if a service definition exists' );
$this -> assertFalse ( $builder -> hasDefinition ( 'foobar' ), '->hasDefinition() returns false if a service definition does not exist' );
$builder -> setDefinition ( 'foobar' , $foo = new Definition ( 'FooBarClass' ));
$this -> assertEquals ( $foo , $builder -> getDefinition ( 'foobar' ), '->getDefinition() returns a service definition if defined' );
$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' );
try {
$builder -> getDefinition ( 'baz' );
$this -> fail ( '->getDefinition() throws a ServiceNotFoundException if the service definition does not exist' );
} catch ( ServiceNotFoundException $e ) {
$this -> assertEquals ( 'You have requested a non-existent service "baz".' , $e -> getMessage (), '->getDefinition() throws a ServiceNotFoundException if the service definition does not exist' );
}
}
/**
* @ group legacy
* @ expectedDeprecation The " deprecated_foo " service is deprecated . You should stop using it , as it will soon be removed .
*/
public function testCreateDeprecatedService ()
{
$definition = new Definition ( 'stdClass' );
$definition -> setDeprecated ( true );
$builder = new ContainerBuilder ();
$builder -> setDefinition ( 'deprecated_foo' , $definition );
$builder -> get ( 'deprecated_foo' );
}
public function testRegister ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'foo' , 'Bar\FooClass' );
$this -> assertTrue ( $builder -> hasDefinition ( 'foo' ), '->register() registers a new service definition' );
$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 ();
$this -> assertFalse ( $builder -> has ( 'foo' ), '->has() returns false if the service does not exist' );
$builder -> register ( 'foo' , 'Bar\FooClass' );
$this -> assertTrue ( $builder -> has ( 'foo' ), '->has() returns true if a service definition exists' );
$builder -> set ( 'bar' , new \stdClass ());
$this -> assertTrue ( $builder -> has ( 'bar' ), '->has() returns true if a service exists' );
}
/**
* @ expectedException \Symfony\Component\DependencyInjection\Exception\ServiceNotFoundException
* @ expectedExceptionMessage You have requested a non - existent service " foo " .
*/
public function testGetThrowsExceptionIfServiceDoesNotExist ()
{
$builder = new ContainerBuilder ();
$builder -> get ( 'foo' );
}
public function testGetReturnsNullIfServiceDoesNotExistAndInvalidReferenceIsUsed ()
{
$builder = new ContainerBuilder ();
$this -> assertNull ( $builder -> get ( 'foo' , ContainerInterface :: NULL_ON_INVALID_REFERENCE ), '->get() returns null if the service does not exist and NULL_ON_INVALID_REFERENCE is passed as a second argument' );
}
/**
* @ expectedException \Symfony\Component\DependencyInjection\Exception\ServiceCircularReferenceException
*/
public function testGetThrowsCircularReferenceExceptionIfServiceHasReferenceToItself ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'baz' , 'stdClass' ) -> setArguments ( array ( new Reference ( 'baz' )));
$builder -> get ( 'baz' );
}
public function testGetReturnsSameInstanceWhenServiceIsShared ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'bar' , 'stdClass' );
$this -> assertTrue ( $builder -> get ( 'bar' ) === $builder -> get ( 'bar' ), '->get() always returns the same instance if the service is shared' );
}
public function testGetCreatesServiceBasedOnDefinition ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'foo' , 'stdClass' );
$this -> assertInternalType ( 'object' , $builder -> get ( 'foo' ), '->get() returns the service definition associated with the id' );
}
public function testGetReturnsRegisteredService ()
{
$builder = new ContainerBuilder ();
$builder -> set ( 'bar' , $bar = new \stdClass ());
$this -> assertSame ( $bar , $builder -> get ( 'bar' ), '->get() returns the service associated with the id' );
}
public function testRegisterDoesNotOverrideExistingService ()
{
$builder = new ContainerBuilder ();
$builder -> set ( 'bar' , $bar = new \stdClass ());
$builder -> register ( 'bar' , 'stdClass' );
$this -> assertSame ( $bar , $builder -> get ( 'bar' ), '->get() returns the service associated with the id even if a definition has been defined' );
}
public function testNonSharedServicesReturnsDifferentInstances ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'bar' , 'stdClass' ) -> setShared ( false );
$this -> assertNotSame ( $builder -> get ( 'bar' ), $builder -> get ( 'bar' ));
}
/**
* @ expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException
* @ expectedExceptionMessage You have requested a synthetic service ( " foo " ) . The DIC does not know how to construct this service .
*/
public function testGetUnsetLoadingServiceWhenCreateServiceThrowsAnException ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'foo' , 'stdClass' ) -> setSynthetic ( true );
// we expect a RuntimeException here as foo is synthetic
try {
$builder -> get ( 'foo' );
} catch ( RuntimeException $e ) {
}
// we must also have the same RuntimeException here
$builder -> get ( 'foo' );
}
public function testGetServiceIds ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'foo' , 'stdClass' );
$builder -> bar = $bar = new \stdClass ();
$builder -> register ( 'bar' , 'stdClass' );
$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 ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'foo' , 'stdClass' );
$builder -> setAlias ( 'bar' , 'foo' );
$this -> assertTrue ( $builder -> hasAlias ( 'bar' ), '->hasAlias() returns true if the alias exists' );
$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 -> assertSame ( $builder -> get ( 'bar' ), $builder -> get ( 'foo' ), '->setAlias() creates a service that is an alias to another one' );
try {
$builder -> setAlias ( 'foobar' , 'foobar' );
$this -> fail ( '->setAlias() throws an InvalidArgumentException if the alias references itself' );
} catch ( \InvalidArgumentException $e ) {
$this -> assertEquals ( 'An alias can not reference itself, got a circular reference on "foobar".' , $e -> getMessage (), '->setAlias() throws an InvalidArgumentException if the alias references itself' );
}
try {
$builder -> getAlias ( 'foobar' );
$this -> fail ( '->getAlias() throws an InvalidArgumentException if the alias does not exist' );
} catch ( \InvalidArgumentException $e ) {
$this -> assertEquals ( 'The service alias "foobar" does not exist.' , $e -> getMessage (), '->getAlias() throws an InvalidArgumentException if the alias does not exist' );
}
}
public function testGetAliases ()
{
$builder = new ContainerBuilder ();
$builder -> setAlias ( 'bar' , 'foo' );
$builder -> setAlias ( 'foobar' , 'foo' );
$builder -> setAlias ( 'moo' , new Alias ( 'foo' , false ));
$aliases = $builder -> getAliases ();
$this -> assertEquals ( 'foo' , ( string ) $aliases [ 'bar' ]);
$this -> assertTrue ( $aliases [ 'bar' ] -> isPublic ());
$this -> assertEquals ( 'foo' , ( string ) $aliases [ 'foobar' ]);
$this -> assertEquals ( 'foo' , ( string ) $aliases [ 'moo' ]);
$this -> assertFalse ( $aliases [ 'moo' ] -> isPublic ());
$builder -> register ( 'bar' , 'stdClass' );
$this -> assertFalse ( $builder -> hasAlias ( 'bar' ));
$builder -> set ( 'foobar' , 'stdClass' );
$builder -> set ( 'moo' , 'stdClass' );
$this -> assertCount ( 2 , $builder -> getAliases (), '->getAliases() does not return aliased services that have been overridden' );
}
public function testSetAliases ()
{
$builder = new ContainerBuilder ();
$builder -> setAliases ( array ( 'bar' => 'foo' , 'foobar' => 'foo' ));
$aliases = $builder -> getAliases ();
$this -> assertArrayHasKey ( 'bar' , $aliases );
$this -> assertArrayHasKey ( 'foobar' , $aliases );
}
public function testAddAliases ()
{
$builder = new ContainerBuilder ();
$builder -> setAliases ( array ( 'bar' => 'foo' ));
$builder -> addAliases ( array ( 'foobar' => 'foo' ));
$aliases = $builder -> getAliases ();
$this -> assertArrayHasKey ( 'bar' , $aliases );
$this -> assertArrayHasKey ( 'foobar' , $aliases );
}
public function testSetReplacesAlias ()
{
$builder = new ContainerBuilder ();
$builder -> setAlias ( 'alias' , 'aliased' );
$builder -> set ( 'aliased' , new \stdClass ());
$builder -> set ( 'alias' , $foo = new \stdClass ());
$this -> assertSame ( $foo , $builder -> get ( 'alias' ), '->set() replaces an existing alias' );
}
public function testAliasesKeepInvalidBehavior ()
{
$builder = new ContainerBuilder ();
$aliased = new Definition ( 'stdClass' );
$aliased -> addMethodCall ( 'setBar' , array ( new Reference ( 'bar' , ContainerInterface :: IGNORE_ON_INVALID_REFERENCE )));
$builder -> setDefinition ( 'aliased' , $aliased );
$builder -> setAlias ( 'alias' , 'aliased' );
$this -> assertEquals ( new \stdClass (), $builder -> get ( 'alias' ));
}
public function testAddGetCompilerPass ()
{
$builder = new ContainerBuilder ();
$builder -> setResourceTracking ( false );
$defaultPasses = $builder -> getCompiler () -> getPassConfig () -> getPasses ();
$builder -> addCompilerPass ( $pass1 = $this -> getMockBuilder ( 'Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface' ) -> getMock (), PassConfig :: TYPE_BEFORE_OPTIMIZATION , - 5 );
$builder -> addCompilerPass ( $pass2 = $this -> getMockBuilder ( 'Symfony\Component\DependencyInjection\Compiler\CompilerPassInterface' ) -> getMock (), PassConfig :: TYPE_BEFORE_OPTIMIZATION , 10 );
$passes = $builder -> getCompiler () -> getPassConfig () -> getPasses ();
$this -> assertCount ( \count ( $passes ) - 2 , $defaultPasses );
// Pass 1 is executed later
$this -> assertTrue ( array_search ( $pass1 , $passes , true ) > array_search ( $pass2 , $passes , true ));
}
public function testCreateService ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'foo1' , 'Bar\FooClass' ) -> setFile ( __DIR__ . '/Fixtures/includes/foo.php' );
$builder -> register ( 'foo2' , 'Bar\FooClass' ) -> setFile ( __DIR__ . '/Fixtures/includes/%file%.php' );
$builder -> setParameter ( 'file' , 'foo' );
$this -> assertInstanceOf ( '\Bar\FooClass' , $builder -> get ( 'foo1' ), '->createService() requires the file defined by the service definition' );
$this -> assertInstanceOf ( '\Bar\FooClass' , $builder -> get ( 'foo2' ), '->createService() replaces parameters in the file provided by the service definition' );
}
public function testCreateProxyWithRealServiceInstantiator ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'foo1' , 'Bar\FooClass' ) -> setFile ( __DIR__ . '/Fixtures/includes/foo.php' );
$builder -> getDefinition ( 'foo1' ) -> setLazy ( true );
$foo1 = $builder -> get ( 'foo1' );
$this -> assertSame ( $foo1 , $builder -> get ( 'foo1' ), 'The same proxy is retrieved on multiple subsequent calls' );
$this -> assertSame ( 'Bar\FooClass' , \get_class ( $foo1 ));
}
public function testCreateServiceClass ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'foo1' , '%class%' );
$builder -> setParameter ( 'class' , 'stdClass' );
$this -> assertInstanceOf ( '\stdClass' , $builder -> get ( 'foo1' ), '->createService() replaces parameters in the class provided by the service definition' );
}
public function testCreateServiceArguments ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'bar' , 'stdClass' );
$builder -> register ( 'foo1' , 'Bar\FooClass' ) -> addArgument ( array ( 'foo' => '%value%' , '%value%' => 'foo' , new Reference ( 'bar' ), '%%unescape_it%%' ));
$builder -> setParameter ( 'value' , 'bar' );
$this -> assertEquals ( array ( 'foo' => 'bar' , 'bar' => 'foo' , $builder -> get ( 'bar' ), '%unescape_it%' ), $builder -> get ( 'foo1' ) -> arguments , '->createService() replaces parameters and service references in the arguments provided by the service definition' );
}
public function testCreateServiceFactory ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'foo' , 'Bar\FooClass' ) -> setFactory ( 'Bar\FooClass::getInstance' );
$builder -> register ( 'qux' , 'Bar\FooClass' ) -> setFactory ( array ( 'Bar\FooClass' , 'getInstance' ));
$builder -> register ( 'bar' , 'Bar\FooClass' ) -> setFactory ( array ( new Definition ( 'Bar\FooClass' ), 'getInstance' ));
$builder -> register ( 'baz' , 'Bar\FooClass' ) -> setFactory ( array ( new Reference ( 'bar' ), 'getInstance' ));
$this -> assertTrue ( $builder -> get ( 'foo' ) -> called , '->createService() calls the factory method to create the service instance' );
$this -> assertTrue ( $builder -> get ( 'qux' ) -> called , '->createService() calls the factory method to create the service instance' );
$this -> assertTrue ( $builder -> get ( 'bar' ) -> called , '->createService() uses anonymous service as factory' );
$this -> assertTrue ( $builder -> get ( 'baz' ) -> called , '->createService() uses another service as factory' );
}
public function testCreateServiceMethodCalls ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'bar' , 'stdClass' );
$builder -> register ( 'foo1' , 'Bar\FooClass' ) -> addMethodCall ( 'setBar' , array ( array ( '%value%' , new Reference ( 'bar' ))));
$builder -> setParameter ( 'value' , 'bar' );
$this -> assertEquals ( array ( 'bar' , $builder -> get ( 'bar' )), $builder -> get ( 'foo1' ) -> bar , '->createService() replaces the values in the method calls arguments' );
}
public function testCreateServiceMethodCallsWithEscapedParam ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'bar' , 'stdClass' );
$builder -> register ( 'foo1' , 'Bar\FooClass' ) -> addMethodCall ( 'setBar' , array ( array ( '%%unescape_it%%' )));
$builder -> setParameter ( 'value' , 'bar' );
$this -> assertEquals ( array ( '%unescape_it%' ), $builder -> get ( 'foo1' ) -> bar , '->createService() replaces the values in the method calls arguments' );
}
public function testCreateServiceProperties ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'bar' , 'stdClass' );
$builder -> register ( 'foo1' , 'Bar\FooClass' ) -> setProperty ( 'bar' , array ( '%value%' , new Reference ( 'bar' ), '%%unescape_it%%' ));
$builder -> setParameter ( 'value' , 'bar' );
$this -> assertEquals ( array ( 'bar' , $builder -> get ( 'bar' ), '%unescape_it%' ), $builder -> get ( 'foo1' ) -> bar , '->createService() replaces the values in the properties' );
}
public function testCreateServiceConfigurator ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'foo1' , 'Bar\FooClass' ) -> setConfigurator ( 'sc_configure' );
$builder -> register ( 'foo2' , 'Bar\FooClass' ) -> setConfigurator ( array ( '%class%' , 'configureStatic' ));
$builder -> setParameter ( 'class' , 'BazClass' );
$builder -> register ( 'baz' , 'BazClass' );
$builder -> register ( 'foo3' , 'Bar\FooClass' ) -> setConfigurator ( array ( new Reference ( 'baz' ), 'configure' ));
$builder -> register ( 'foo4' , 'Bar\FooClass' ) -> setConfigurator ( array ( $builder -> getDefinition ( 'baz' ), 'configure' ));
$builder -> register ( 'foo5' , 'Bar\FooClass' ) -> setConfigurator ( 'foo' );
$this -> assertTrue ( $builder -> get ( 'foo1' ) -> configured , '->createService() calls the configurator' );
$this -> assertTrue ( $builder -> get ( 'foo2' ) -> configured , '->createService() calls the configurator' );
$this -> assertTrue ( $builder -> get ( 'foo3' ) -> configured , '->createService() calls the configurator' );
$this -> assertTrue ( $builder -> get ( 'foo4' ) -> configured , '->createService() calls the configurator' );
try {
$builder -> get ( 'foo5' );
$this -> fail ( '->createService() throws an InvalidArgumentException if the configure callable is not a valid callable' );
} catch ( \InvalidArgumentException $e ) {
$this -> assertEquals ( 'The configure callable for class "Bar\FooClass" is not a callable.' , $e -> getMessage (), '->createService() throws an InvalidArgumentException if the configure callable is not a valid callable' );
}
}
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
*/
public function testCreateSyntheticService ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'foo' , 'Bar\FooClass' ) -> setSynthetic ( true );
$builder -> get ( 'foo' );
}
public function testCreateServiceWithExpression ()
{
$builder = new ContainerBuilder ();
$builder -> setParameter ( 'bar' , 'bar' );
$builder -> register ( 'bar' , 'BarClass' );
$builder -> register ( 'foo' , 'Bar\FooClass' ) -> addArgument ( array ( 'foo' => new Expression ( 'service("bar").foo ~ parameter("bar")' )));
$this -> assertEquals ( 'foobar' , $builder -> get ( 'foo' ) -> arguments [ 'foo' ]);
}
public function testResolveServices ()
{
$builder = new ContainerBuilder ();
$builder -> register ( 'foo' , 'Bar\FooClass' );
$this -> assertEquals ( $builder -> get ( 'foo' ), $builder -> resolveServices ( new Reference ( 'foo' )), '->resolveServices() resolves service references to service instances' );
$this -> assertEquals ( array ( 'foo' => array ( 'foo' , $builder -> get ( 'foo' ))), $builder -> resolveServices ( array ( 'foo' => array ( 'foo' , new Reference ( 'foo' )))), '->resolveServices() resolves service references to service instances in nested arrays' );
$this -> assertEquals ( $builder -> get ( 'foo' ), $builder -> resolveServices ( new Expression ( 'service("foo")' )), '->resolveServices() resolves expressions' );
}
/**
* @ expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException
* @ expectedExceptionMessage Constructing service " foo " from a parent definition is not supported at build time .
*/
public function testResolveServicesWithDecoratedDefinition ()
{
$builder = new ContainerBuilder ();
$builder -> setDefinition ( 'grandpa' , new Definition ( 'stdClass' ));
$builder -> setDefinition ( 'parent' , new ChildDefinition ( 'grandpa' ));
$builder -> setDefinition ( 'foo' , new ChildDefinition ( 'parent' ));
$builder -> get ( 'foo' );
}
public function testResolveServicesWithCustomDefinitionClass ()
{
$builder = new ContainerBuilder ();
$builder -> setDefinition ( 'foo' , new CustomDefinition ( 'stdClass' ));
$this -> assertInstanceOf ( 'stdClass' , $builder -> get ( 'foo' ));
}
public function testMerge ()
{
$container = new ContainerBuilder ( new ParameterBag ( array ( 'bar' => 'foo' )));
$container -> setResourceTracking ( false );
$config = new ContainerBuilder ( new ParameterBag ( array ( 'foo' => 'bar' )));
$container -> merge ( $config );
$this -> assertEquals ( array ( 'bar' => 'foo' , 'foo' => 'bar' ), $container -> getParameterBag () -> all (), '->merge() merges current parameters with the loaded ones' );
$container = new ContainerBuilder ( new ParameterBag ( array ( 'bar' => 'foo' )));
$container -> setResourceTracking ( false );
$config = new ContainerBuilder ( new ParameterBag ( array ( 'foo' => '%bar%' )));
$container -> merge ( $config );
$container -> compile ();
$this -> assertEquals ( array ( 'bar' => 'foo' , 'foo' => 'foo' ), $container -> getParameterBag () -> all (), '->merge() evaluates the values of the parameters towards already defined ones' );
$container = new ContainerBuilder ( new ParameterBag ( array ( 'bar' => 'foo' )));
$container -> setResourceTracking ( false );
$config = new ContainerBuilder ( new ParameterBag ( array ( 'foo' => '%bar%' , 'baz' => '%foo%' )));
$container -> merge ( $config );
$container -> compile ();
$this -> assertEquals ( array ( 'bar' => 'foo' , 'foo' => 'foo' , 'baz' => 'foo' ), $container -> getParameterBag () -> all (), '->merge() evaluates the values of the parameters towards already defined ones' );
$container = new ContainerBuilder ();
$container -> setResourceTracking ( false );
$container -> register ( 'foo' , 'Bar\FooClass' );
$container -> register ( 'bar' , 'BarClass' );
$config = new ContainerBuilder ();
$config -> setDefinition ( 'baz' , new Definition ( 'BazClass' ));
$config -> setAlias ( 'alias_for_foo' , 'foo' );
$container -> merge ( $config );
2019-01-24 08:00:03 +00:00
$this -> assertEquals ( array ( 'foo' , 'bar' , 'service_container' , 'baz' ), array_keys ( $container -> getDefinitions ()), '->merge() merges definitions already defined ones' );
2018-11-23 12:29:20 +00:00
$aliases = $container -> getAliases ();
$this -> assertArrayHasKey ( 'alias_for_foo' , $aliases );
$this -> assertEquals ( 'foo' , ( string ) $aliases [ 'alias_for_foo' ]);
$container = new ContainerBuilder ();
$container -> setResourceTracking ( false );
$container -> register ( 'foo' , 'Bar\FooClass' );
$config -> setDefinition ( 'foo' , new Definition ( 'BazClass' ));
$container -> merge ( $config );
$this -> assertEquals ( 'BazClass' , $container -> getDefinition ( 'foo' ) -> getClass (), '->merge() overrides already defined services' );
$container = new ContainerBuilder ();
$bag = new EnvPlaceholderParameterBag ();
$bag -> get ( 'env(Foo)' );
$config = new ContainerBuilder ( $bag );
$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' );
}
public function testCompileWithArrayResolveEnv ()
{
putenv ( 'ARRAY={"foo":"bar"}' );
$container = new ContainerBuilder ();
$container -> setParameter ( 'foo' , '%env(json:ARRAY)%' );
$container -> compile ( true );
$this -> assertSame ( array ( 'foo' => 'bar' ), $container -> getParameter ( 'foo' ));
putenv ( 'ARRAY' );
}
public function testCompileWithArrayAndAnotherResolveEnv ()
{
putenv ( 'DUMMY_ENV_VAR=abc' );
putenv ( 'ARRAY={"foo":"bar"}' );
$container = new ContainerBuilder ();
$container -> setParameter ( 'foo' , '%env(json:ARRAY)%' );
$container -> setParameter ( 'bar' , '%env(DUMMY_ENV_VAR)%' );
$container -> compile ( true );
$this -> assertSame ( array ( 'foo' => 'bar' ), $container -> getParameter ( 'foo' ));
$this -> assertSame ( 'abc' , $container -> getParameter ( 'bar' ));
putenv ( 'DUMMY_ENV_VAR' );
putenv ( 'ARRAY' );
}
/**
* @ expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException
* @ expectedExceptionMessage A string value must be composed of strings and / or numbers , but found parameter " env(json:ARRAY) " of type array inside string value " ABC %env(json:ARRAY)% " .
*/
public function testCompileWithArrayInStringResolveEnv ()
{
putenv ( 'ARRAY={"foo":"bar"}' );
$container = new ContainerBuilder ();
$container -> setParameter ( 'foo' , 'ABC %env(json:ARRAY)%' );
$container -> compile ( true );
putenv ( 'ARRAY' );
}
/**
* @ 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' );
}
}
/**
* @ expectedException \LogicException
*/
public function testMergeLogicException ()
{
$container = new ContainerBuilder ();
$container -> setResourceTracking ( false );
$container -> compile ();
$container -> merge ( new ContainerBuilder ());
}
public function testfindTaggedServiceIds ()
{
$builder = new ContainerBuilder ();
$builder
-> register ( 'foo' , 'Bar\FooClass' )
-> addTag ( 'foo' , array ( 'foo' => 'foo' ))
-> addTag ( 'bar' , array ( 'bar' => 'bar' ))
-> addTag ( 'foo' , array ( 'foofoo' => 'foofoo' ))
;
$this -> assertEquals ( $builder -> findTaggedServiceIds ( 'foo' ), array (
'foo' => array (
array ( 'foo' => 'foo' ),
array ( 'foofoo' => 'foofoo' ),
),
), '->findTaggedServiceIds() returns an array of service ids and its tag attributes' );
$this -> assertEquals ( array (), $builder -> findTaggedServiceIds ( 'foobar' ), '->findTaggedServiceIds() returns an empty array if there is annotated services' );
}
public function testFindUnusedTags ()
{
$builder = new ContainerBuilder ();
$builder
-> register ( 'foo' , 'Bar\FooClass' )
-> addTag ( 'kernel.event_listener' , array ( 'foo' => 'foo' ))
-> addTag ( 'kenrel.event_listener' , array ( 'bar' => 'bar' ))
;
$builder -> findTaggedServiceIds ( 'kernel.event_listener' );
$this -> assertEquals ( array ( 'kenrel.event_listener' ), $builder -> findUnusedTags (), '->findUnusedTags() returns an array with unused tags' );
}
public function testFindDefinition ()
{
$container = new ContainerBuilder ();
$container -> setDefinition ( 'foo' , $definition = new Definition ( 'Bar\FooClass' ));
$container -> setAlias ( 'bar' , 'foo' );
$container -> setAlias ( 'foobar' , 'bar' );
$this -> assertEquals ( $definition , $container -> findDefinition ( 'foobar' ), '->findDefinition() returns a Definition' );
}
public function testAddObjectResource ()
{
$container = new ContainerBuilder ();
$container -> setResourceTracking ( false );
$container -> addObjectResource ( new \BarClass ());
$this -> assertEmpty ( $container -> getResources (), 'No resources get registered without resource tracking' );
$container -> setResourceTracking ( true );
$container -> addObjectResource ( new \BarClass ());
$resources = $container -> getResources ();
$this -> assertCount ( 2 , $resources , '2 resources were registered' );
/* @var $resource \Symfony\Component\Config\Resource\FileResource */
$resource = end ( $resources );
$this -> assertInstanceOf ( 'Symfony\Component\Config\Resource\FileResource' , $resource );
$this -> assertSame ( realpath ( __DIR__ . '/Fixtures/includes/classes.php' ), realpath ( $resource -> getResource ()));
}
/**
* @ group legacy
*/
public function testAddClassResource ()
{
$container = new ContainerBuilder ();
$container -> setResourceTracking ( false );
$container -> addClassResource ( new \ReflectionClass ( 'BarClass' ));
$this -> assertEmpty ( $container -> getResources (), 'No resources get registered without resource tracking' );
$container -> setResourceTracking ( true );
$container -> addClassResource ( new \ReflectionClass ( 'BarClass' ));
$resources = $container -> getResources ();
$this -> assertCount ( 2 , $resources , '2 resources were registered' );
/* @var $resource \Symfony\Component\Config\Resource\FileResource */
$resource = end ( $resources );
$this -> assertInstanceOf ( 'Symfony\Component\Config\Resource\FileResource' , $resource );
$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' ) -> setPublic ( true );
$container -> getDefinition ( 'foo' ) -> setLazy ( true );
$container -> compile ();
$matchingResources = array_filter (
$container -> getResources (),
function ( ResourceInterface $resource ) {
return 'reflection.BarClass' === ( string ) $resource ;
}
);
$this -> assertNotEmpty ( $matchingResources );
}
public function testResources ()
{
$container = new ContainerBuilder ();
$container -> addResource ( $a = new FileResource ( __DIR__ . '/Fixtures/xml/services1.xml' ));
$container -> addResource ( $b = new FileResource ( __DIR__ . '/Fixtures/xml/services2.xml' ));
$resources = array ();
foreach ( $container -> getResources () as $resource ) {
if ( false === strpos ( $resource , '.php' )) {
$resources [] = $resource ;
}
}
$this -> assertEquals ( array ( $a , $b ), $resources , '->getResources() returns an array of resources read for the current configuration' );
$this -> assertSame ( $container , $container -> setResources ( array ()));
$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 -> assertSame ( $container -> getExtension ( 'project' ), $extension , '->registerExtension() registers an extension' );
$this -> { method_exists ( $this , $_ = 'expectException' ) ? $_ : 'setExpectedException' }( 'LogicException' );
$container -> getExtension ( 'no_registered' );
}
public function testRegisteredButNotLoadedExtension ()
{
$extension = $this -> getMockBuilder ( 'Symfony\\Component\\DependencyInjection\\Extension\\ExtensionInterface' ) -> getMock ();
$extension -> expects ( $this -> once ()) -> method ( 'getAlias' ) -> will ( $this -> returnValue ( 'project' ));
$extension -> expects ( $this -> never ()) -> method ( 'load' );
$container = new ContainerBuilder ();
$container -> setResourceTracking ( false );
$container -> registerExtension ( $extension );
$container -> compile ();
}
public function testRegisteredAndLoadedExtension ()
{
$extension = $this -> getMockBuilder ( 'Symfony\\Component\\DependencyInjection\\Extension\\ExtensionInterface' ) -> getMock ();
$extension -> expects ( $this -> exactly ( 2 )) -> method ( 'getAlias' ) -> will ( $this -> returnValue ( 'project' ));
$extension -> expects ( $this -> once ()) -> method ( 'load' ) -> with ( array ( array ( 'foo' => 'bar' )));
$container = new ContainerBuilder ();
$container -> setResourceTracking ( false );
$container -> registerExtension ( $extension );
$container -> loadFromExtension ( 'project' , array ( 'foo' => 'bar' ));
$container -> compile ();
}
public function testPrivateServiceUser ()
{
$fooDefinition = new Definition ( 'BarClass' );
$fooUserDefinition = new Definition ( 'BarUserClass' , array ( new Reference ( 'bar' )));
$container = new ContainerBuilder ();
$container -> setResourceTracking ( false );
$fooDefinition -> setPublic ( false );
$container -> addDefinitions ( array (
'bar' => $fooDefinition ,
'bar_user' => $fooUserDefinition -> setPublic ( true ),
));
$container -> compile ();
$this -> assertInstanceOf ( 'BarClass' , $container -> get ( 'bar_user' ) -> bar );
}
/**
* @ expectedException \BadMethodCallException
*/
public function testThrowsExceptionWhenSetServiceOnACompiledContainer ()
{
$container = new ContainerBuilder ();
$container -> setResourceTracking ( false );
$container -> register ( 'a' , 'stdClass' ) -> setPublic ( true );
$container -> compile ();
$container -> set ( 'a' , new \stdClass ());
}
public function testThrowsExceptionWhenAddServiceOnACompiledContainer ()
{
$container = new ContainerBuilder ();
$container -> compile ();
$container -> set ( 'a' , $foo = new \stdClass ());
$this -> assertSame ( $foo , $container -> get ( 'a' ));
}
public function testNoExceptionWhenSetSyntheticServiceOnACompiledContainer ()
{
$container = new ContainerBuilder ();
$def = new Definition ( 'stdClass' );
$def -> setSynthetic ( true ) -> setPublic ( true );
$container -> setDefinition ( 'a' , $def );
$container -> compile ();
$container -> set ( 'a' , $a = new \stdClass ());
$this -> assertEquals ( $a , $container -> get ( 'a' ));
}
/**
* @ expectedException \BadMethodCallException
*/
public function testThrowsExceptionWhenSetDefinitionOnACompiledContainer ()
{
$container = new ContainerBuilder ();
$container -> setResourceTracking ( false );
$container -> compile ();
$container -> setDefinition ( 'a' , new Definition ());
}
public function testExtensionConfig ()
{
$container = new ContainerBuilder ();
$configs = $container -> getExtensionConfig ( 'foo' );
$this -> assertEmpty ( $configs );
$first = array ( 'foo' => 'bar' );
$container -> prependExtensionConfig ( 'foo' , $first );
$configs = $container -> getExtensionConfig ( 'foo' );
$this -> assertEquals ( array ( $first ), $configs );
$second = array ( 'ding' => 'dong' );
$container -> prependExtensionConfig ( 'foo' , $second );
$configs = $container -> getExtensionConfig ( 'foo' );
$this -> assertEquals ( array ( $second , $first ), $configs );
}
public function testAbstractAlias ()
{
$container = new ContainerBuilder ();
$abstract = new Definition ( 'AbstractClass' );
$abstract -> setAbstract ( true ) -> setPublic ( true );
$container -> setDefinition ( 'abstract_service' , $abstract );
$container -> setAlias ( 'abstract_alias' , 'abstract_service' ) -> setPublic ( true );
$container -> compile ();
$this -> assertSame ( 'abstract_service' , ( string ) $container -> getAlias ( 'abstract_alias' ));
}
public function testLazyLoadedService ()
{
$loader = new ClosureLoader ( $container = new ContainerBuilder ());
$loader -> load ( function ( ContainerBuilder $container ) {
$container -> set ( 'a' , new \BazClass ());
$definition = new Definition ( 'BazClass' );
$definition -> setLazy ( true );
$definition -> setPublic ( true );
$container -> setDefinition ( 'a' , $definition );
});
$container -> setResourceTracking ( true );
$container -> compile ();
$r = new \ReflectionProperty ( $container , 'resources' );
$r -> setAccessible ( true );
$resources = $r -> getValue ( $container );
$classInList = false ;
foreach ( $resources as $resource ) {
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 \T estClass -> 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 -> compile ();
$this -> assertTrue ( $container -> get ( 'bar' ) -> callPassed (), '->compile() initializes properties before method calls' );
}
public function testAutowiring ()
{
$container = new ContainerBuilder ();
$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 ();
}
/**
* @ 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 testNoClassFromGlobalNamespaceClassIdWithLeadingSlash ()
{
$container = new ContainerBuilder ();
$container -> register ( '\\' . \DateTime :: class );
$container -> compile ();
}
/**
* @ expectedException \Symfony\Component\DependencyInjection\Exception\RuntimeException
* @ expectedExceptionMessage The definition for " \ Symfony \ Component \ DependencyInjection \T ests \ FooClass " has no class attribute , and appears to reference a class or interface . Please specify the class attribute explicitly or remove the leading backslash by renaming the service to " Symfony \ Component \ DependencyInjection \T ests \ FooClass " to get rid of this error .
*/
public function testNoClassFromNamespaceClassIdWithLeadingSlash ()
{
$container = new ContainerBuilder ();
$container -> register ( '\\' . FooClass :: class );
$container -> compile ();
}
/**
* @ 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 );
$manager = $container -> get ( 'manager' );
$this -> assertEquals ( new \stdClass (), $manager );
$manager = $container -> get ( 'manager2' );
$this -> assertEquals ( new \stdClass (), $manager );
$foo6 = $container -> get ( 'foo6' );
$this -> assertEquals (( object ) array ( 'bar6' => ( object ) array ()), $foo6 );
2019-01-24 08:00:03 +00:00
$this -> assertInstanceOf ( \stdClass :: class , $container -> get ( 'root' ));
2018-11-23 12:29:20 +00:00
}
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' ));
}
/**
* This test checks the trigger of a deprecation note and should not be removed in major releases .
*
* @ group legacy
* @ expectedDeprecation The " foo " service is deprecated . You should stop using it , as it will soon be removed .
*/
public function testPrivateServiceTriggersDeprecation ()
{
$container = new ContainerBuilder ();
$container -> register ( 'foo' , 'stdClass' )
-> setPublic ( false )
-> setDeprecated ( true );
$container -> register ( 'bar' , 'stdClass' )
-> setPublic ( true )
-> setProperty ( 'foo' , new Reference ( 'foo' ));
$container -> compile ();
$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 );
}
public function testUninitializedSyntheticReference ()
{
$container = new ContainerBuilder ();
$container -> register ( 'foo' , 'stdClass' ) -> setPublic ( true ) -> setSynthetic ( true );
$container -> register ( 'bar' , 'stdClass' ) -> setPublic ( true ) -> setShared ( false )
-> setProperty ( 'foo' , new Reference ( 'foo' , ContainerInterface :: IGNORE_ON_UNINITIALIZED_REFERENCE ));
$container -> compile ();
$this -> assertEquals (( object ) array ( 'foo' => null ), $container -> get ( 'bar' ));
$container -> set ( 'foo' , ( object ) array ( 123 ));
$this -> assertEquals (( object ) array ( 'foo' => ( object ) array ( 123 )), $container -> get ( 'bar' ));
}
public function testDecoratedSelfReferenceInvolvingPrivateServices ()
{
$container = new ContainerBuilder ();
$container -> register ( 'foo' , 'stdClass' )
-> setPublic ( false )
-> setProperty ( 'bar' , new Reference ( 'foo' ));
$container -> register ( 'baz' , 'stdClass' )
-> setPublic ( false )
-> setProperty ( 'inner' , new Reference ( 'baz.inner' ))
-> setDecoratedService ( 'foo' );
$container -> compile ();
$this -> assertSame ( array ( 'service_container' ), array_keys ( $container -> getDefinitions ()));
}
}
class FooClass
{
}
class A
{
}
class B
{
public function __construct ( A $a )
{
}
}