ContainerBuilderTest.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399
  1. <?php
  2. /*
  3. * This file is part of the Symfony package.
  4. * (c) Fabien Potencier <fabien.potencier@symfony-project.com>
  5. *
  6. * For the full copyright and license information, please view the LICENSE
  7. * file that was distributed with this source code.
  8. */
  9. namespace Symfony\Tests\Component\DependencyInjection;
  10. use Symfony\Component\DependencyInjection\ContainerBuilder;
  11. use Symfony\Component\DependencyInjection\ContainerInterface;
  12. use Symfony\Component\DependencyInjection\Definition;
  13. use Symfony\Component\DependencyInjection\Reference;
  14. use Symfony\Component\DependencyInjection\ParameterBag\ParameterBag;
  15. use Symfony\Component\DependencyInjection\Resource\FileResource;
  16. require_once __DIR__.'/Fixtures/includes/classes.php';
  17. class ContainerBuilderTest extends \PHPUnit_Framework_TestCase
  18. {
  19. /**
  20. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::setDefinitions
  21. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::getDefinitions
  22. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::setDefinition
  23. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::getDefinition
  24. */
  25. public function testDefinitions()
  26. {
  27. $builder = new ContainerBuilder();
  28. $definitions = array(
  29. 'foo' => new Definition('FooClass'),
  30. 'bar' => new Definition('BarClass'),
  31. );
  32. $builder->setDefinitions($definitions);
  33. $this->assertEquals($definitions, $builder->getDefinitions(), '->setDefinitions() sets the service definitions');
  34. $this->assertTrue($builder->hasDefinition('foo'), '->hasDefinition() returns true if a service definition exists');
  35. $this->assertFalse($builder->hasDefinition('foobar'), '->hasDefinition() returns false if a service definition does not exist');
  36. $builder->setDefinition('foobar', $foo = new Definition('FooBarClass'));
  37. $this->assertEquals($foo, $builder->getDefinition('foobar'), '->getDefinition() returns a service definition if defined');
  38. $this->assertTrue($builder->setDefinition('foobar', $foo = new Definition('FooBarClass')) === $foo, '->setDefinition() implements a fuild interface by returning the service reference');
  39. $builder->addDefinitions($defs = array('foobar' => new Definition('FooBarClass')));
  40. $this->assertEquals(array_merge($definitions, $defs), $builder->getDefinitions(), '->addDefinitions() adds the service definitions');
  41. try {
  42. $builder->getDefinition('baz');
  43. $this->fail('->getDefinition() throws an InvalidArgumentException if the service definition does not exist');
  44. } catch (\Exception $e) {
  45. $this->assertInstanceOf('\InvalidArgumentException', $e, '->getDefinition() throws an InvalidArgumentException if the service definition does not exist');
  46. $this->assertEquals('The service definition "baz" does not exist.', $e->getMessage(), '->getDefinition() throws an InvalidArgumentException if the service definition does not exist');
  47. }
  48. }
  49. /**
  50. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::register
  51. */
  52. public function testRegister()
  53. {
  54. $builder = new ContainerBuilder();
  55. $builder->register('foo', 'FooClass');
  56. $this->assertTrue($builder->hasDefinition('foo'), '->register() registers a new service definition');
  57. $this->assertInstanceOf('Symfony\Component\DependencyInjection\Definition', $builder->getDefinition('foo'), '->register() returns the newly created Definition instance');
  58. }
  59. /**
  60. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::has
  61. */
  62. public function testHas()
  63. {
  64. $builder = new ContainerBuilder();
  65. $this->assertFalse($builder->has('foo'), '->has() returns false if the service does not exist');
  66. $builder->register('foo', 'FooClass');
  67. $this->assertTrue($builder->has('foo'), '->has() returns true if a service definition exists');
  68. $builder->set('bar', new \stdClass());
  69. $this->assertTrue($builder->has('bar'), '->has() returns true if a service exists');
  70. }
  71. /**
  72. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::get
  73. */
  74. public function testGet()
  75. {
  76. $builder = new ContainerBuilder();
  77. try {
  78. $builder->get('foo');
  79. $this->fail('->get() throws an InvalidArgumentException if the service does not exist');
  80. } catch (\Exception $e) {
  81. $this->assertInstanceOf('\InvalidArgumentException', $e, '->get() throws an InvalidArgumentException if the service does not exist');
  82. $this->assertEquals('The service definition "foo" does not exist.', $e->getMessage(), '->get() throws an InvalidArgumentException if the service does not exist');
  83. }
  84. $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');
  85. $builder->register('foo', 'stdClass');
  86. $this->assertType('object', $builder->get('foo'), '->get() returns the service definition associated with the id');
  87. $builder->set('bar', $bar = new \stdClass());
  88. $this->assertEquals($bar, $builder->get('bar'), '->get() returns the service associated with the id');
  89. $builder->register('bar', 'stdClass');
  90. $this->assertEquals($bar, $builder->get('bar'), '->get() returns the service associated with the id even if a definition has been defined');
  91. $builder->register('baz', 'stdClass')->setArguments(array(new Reference('baz')));
  92. try {
  93. @$builder->get('baz');
  94. $this->fail('->get() throws a LogicException if the service has a circular reference to itself');
  95. } catch (\Exception $e) {
  96. $this->assertInstanceOf('\LogicException', $e, '->get() throws a LogicException if the service has a circular reference to itself');
  97. $this->assertEquals('The service "baz" has a circular reference to itself.', $e->getMessage(), '->get() throws a LogicException if the service has a circular reference to itself');
  98. }
  99. $builder->register('foobar', 'stdClass')->setShared(true);
  100. $this->assertTrue($builder->get('bar') === $builder->get('bar'), '->get() always returns the same instance if the service is shared');
  101. }
  102. /**
  103. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::getServiceIds
  104. */
  105. public function testGetServiceIds()
  106. {
  107. $builder = new ContainerBuilder();
  108. $builder->register('foo', 'stdClass');
  109. $builder->bar = $bar = new \stdClass();
  110. $builder->register('bar', 'stdClass');
  111. $this->assertEquals(array('foo', 'bar', 'service_container'), $builder->getServiceIds(), '->getServiceIds() returns all defined service ids');
  112. }
  113. /**
  114. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::setAlias
  115. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::hasAlias
  116. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::getAlias
  117. */
  118. public function testAliases()
  119. {
  120. $builder = new ContainerBuilder();
  121. $builder->register('foo', 'stdClass');
  122. $builder->setAlias('bar', 'foo');
  123. $this->assertTrue($builder->hasAlias('bar'), '->hasAlias() returns true if the alias exists');
  124. $this->assertFalse($builder->hasAlias('foobar'), '->hasAlias() returns false if the alias does not exist');
  125. $this->assertEquals('foo', $builder->getAlias('bar'), '->getAlias() returns the aliased service');
  126. $this->assertTrue($builder->has('bar'), '->setAlias() defines a new service');
  127. $this->assertTrue($builder->get('bar') === $builder->get('foo'), '->setAlias() creates a service that is an alias to another one');
  128. try {
  129. $builder->getAlias('foobar');
  130. $this->fail('->getAlias() throws an InvalidArgumentException if the alias does not exist');
  131. } catch (\Exception $e) {
  132. $this->assertInstanceOf('\InvalidArgumentException', $e, '->getAlias() throws an InvalidArgumentException if the alias does not exist');
  133. $this->assertEquals('The service alias "foobar" does not exist.', $e->getMessage(), '->getAlias() throws an InvalidArgumentException if the alias does not exist');
  134. }
  135. }
  136. /**
  137. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::getAliases
  138. */
  139. public function testGetAliases()
  140. {
  141. $builder = new ContainerBuilder();
  142. $builder->setAlias('bar', 'foo');
  143. $builder->setAlias('foobar', 'foo');
  144. $this->assertEquals(array('bar' => 'foo', 'foobar' => 'foo'), $builder->getAliases(), '->getAliases() returns all service aliases');
  145. $builder->register('bar', 'stdClass');
  146. $this->assertEquals(array('foobar' => 'foo'), $builder->getAliases(), '->getAliases() does not return aliased services that have been overridden');
  147. $builder->set('foobar', 'stdClass');
  148. $this->assertEquals(array(), $builder->getAliases(), '->getAliases() does not return aliased services that have been overridden');
  149. }
  150. /**
  151. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::setAliases
  152. */
  153. public function testSetAliases()
  154. {
  155. $builder = new ContainerBuilder();
  156. $builder->setAliases(array('bar' => 'foo', 'foobar' => 'foo'));
  157. $this->assertEquals(array('bar' => 'foo', 'foobar' => 'foo'), $builder->getAliases(), '->getAliases() returns all service aliases');
  158. }
  159. /**
  160. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::addAliases
  161. */
  162. public function testAddAliases()
  163. {
  164. $builder = new ContainerBuilder();
  165. $builder->setAliases(array('bar' => 'foo'));
  166. $builder->addAliases(array('foobar' => 'foo'));
  167. $this->assertEquals(array('bar' => 'foo', 'foobar' => 'foo'), $builder->getAliases(), '->getAliases() returns all service aliases');
  168. }
  169. /**
  170. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::createService
  171. */
  172. public function testCreateService()
  173. {
  174. $builder = new ContainerBuilder();
  175. $builder->register('foo1', 'FooClass')->setFile(__DIR__.'/Fixtures/includes/foo.php');
  176. $this->assertInstanceOf('\FooClass', $builder->get('foo1'), '->createService() requires the file defined by the service definition');
  177. $builder->register('foo2', 'FooClass')->setFile(__DIR__.'/Fixtures/includes/%file%.php');
  178. $builder->setParameter('file', 'foo');
  179. $this->assertInstanceOf('\FooClass', $builder->get('foo2'), '->createService() replaces parameters in the file provided by the service definition');
  180. }
  181. /**
  182. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::createService
  183. */
  184. public function testCreateServiceClass()
  185. {
  186. $builder = new ContainerBuilder();
  187. $builder->register('foo1', '%class%');
  188. $builder->setParameter('class', 'stdClass');
  189. $this->assertInstanceOf('\stdClass', $builder->get('foo1'), '->createService() replaces parameters in the class provided by the service definition');
  190. }
  191. /**
  192. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::createService
  193. */
  194. public function testCreateServiceArguments()
  195. {
  196. $builder = new ContainerBuilder();
  197. $builder->register('bar', 'stdClass');
  198. $builder->register('foo1', 'FooClass')->addArgument(array('foo' => '%value%', '%value%' => 'foo', new Reference('bar')));
  199. $builder->setParameter('value', 'bar');
  200. $this->assertEquals(array('foo' => 'bar', 'bar' => 'foo', $builder->get('bar')), $builder->get('foo1')->arguments, '->createService() replaces parameters and service references in the arguments provided by the service definition');
  201. }
  202. /**
  203. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::createService
  204. */
  205. public function testCreateServiceFactoryMethod()
  206. {
  207. $builder = new ContainerBuilder();
  208. $builder->register('bar', 'stdClass');
  209. $builder->register('foo1', 'FooClass')->setFactoryMethod('getInstance')->addArgument(array('foo' => '%value%', '%value%' => 'foo', new Reference('bar')));
  210. $builder->setParameter('value', 'bar');
  211. $this->assertTrue($builder->get('foo1')->called, '->createService() calls the factory method to create the service instance');
  212. $this->assertEquals(array('foo' => 'bar', 'bar' => 'foo', $builder->get('bar')), $builder->get('foo1')->arguments, '->createService() passes the arguments to the factory method');
  213. }
  214. /**
  215. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::createService
  216. */
  217. public function testCreateServiceFactoryService()
  218. {
  219. $builder = new ContainerBuilder();
  220. $builder->register('baz_service')->setFactoryService('baz_factory')->setFactoryMethod('getInstance');
  221. $builder->register('baz_factory', 'BazClass');
  222. $this->assertType('BazClass', $builder->get('baz_service'));
  223. }
  224. /**
  225. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::createService
  226. */
  227. public function testCreateServiceMethodCalls()
  228. {
  229. $builder = new ContainerBuilder();
  230. $builder->register('bar', 'stdClass');
  231. $builder->register('foo1', 'FooClass')->addMethodCall('setBar', array(array('%value%', new Reference('bar'))));
  232. $builder->setParameter('value', 'bar');
  233. $this->assertEquals(array('bar', $builder->get('bar')), $builder->get('foo1')->bar, '->createService() replaces the values in the method calls arguments');
  234. }
  235. /**
  236. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::createService
  237. */
  238. public function testCreateServiceConfigurator()
  239. {
  240. $builder = new ContainerBuilder();
  241. $builder->register('foo1', 'FooClass')->setConfigurator('sc_configure');
  242. $this->assertTrue($builder->get('foo1')->configured, '->createService() calls the configurator');
  243. $builder->register('foo2', 'FooClass')->setConfigurator(array('%class%', 'configureStatic'));
  244. $builder->setParameter('class', 'BazClass');
  245. $this->assertTrue($builder->get('foo2')->configured, '->createService() calls the configurator');
  246. $builder->register('baz', 'BazClass');
  247. $builder->register('foo3', 'FooClass')->setConfigurator(array(new Reference('baz'), 'configure'));
  248. $this->assertTrue($builder->get('foo3')->configured, '->createService() calls the configurator');
  249. $builder->register('foo4', 'FooClass')->setConfigurator('foo');
  250. try {
  251. $builder->get('foo4');
  252. $this->fail('->createService() throws an InvalidArgumentException if the configure callable is not a valid callable');
  253. } catch (\Exception $e) {
  254. $this->assertInstanceOf('\InvalidArgumentException', $e, '->createService() throws an InvalidArgumentException if the configure callable is not a valid callable');
  255. $this->assertEquals('The configure callable for class "FooClass" is not a callable.', $e->getMessage(), '->createService() throws an InvalidArgumentException if the configure callable is not a valid callable');
  256. }
  257. }
  258. /**
  259. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::resolveServices
  260. */
  261. public function testResolveServices()
  262. {
  263. $builder = new ContainerBuilder();
  264. $builder->register('foo', 'FooClass');
  265. $this->assertEquals($builder->get('foo'), $builder->resolveServices(new Reference('foo')), '->resolveServices() resolves service references to service instances');
  266. $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');
  267. }
  268. /**
  269. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::merge
  270. */
  271. public function testMerge()
  272. {
  273. $container = new ContainerBuilder(new ParameterBag(array('bar' => 'foo')));
  274. $config = new ContainerBuilder(new ParameterBag(array('foo' => 'bar')));
  275. $container->merge($config);
  276. $this->assertEquals(array('bar' => 'foo', 'foo' => 'bar'), $container->getParameterBag()->all(), '->merge() merges current parameters with the loaded ones');
  277. $container = new ContainerBuilder(new ParameterBag(array('bar' => 'foo')));
  278. $config = new ContainerBuilder(new ParameterBag(array('foo' => '%bar%')));
  279. $container->merge($config);
  280. ////// FIXME
  281. $container->freeze();
  282. $this->assertEquals(array('bar' => 'foo', 'foo' => 'foo'), $container->getParameterBag()->all(), '->merge() evaluates the values of the parameters towards already defined ones');
  283. $container = new ContainerBuilder(new ParameterBag(array('bar' => 'foo')));
  284. $config = new ContainerBuilder(new ParameterBag(array('foo' => '%bar%', 'baz' => '%foo%')));
  285. $container->merge($config);
  286. ////// FIXME
  287. $container->freeze();
  288. $this->assertEquals(array('bar' => 'foo', 'foo' => 'foo', 'baz' => 'foo'), $container->getParameterBag()->all(), '->merge() evaluates the values of the parameters towards already defined ones');
  289. $container = new ContainerBuilder();
  290. $container->register('foo', 'FooClass');
  291. $container->register('bar', 'BarClass');
  292. $config = new ContainerBuilder();
  293. $config->setDefinition('baz', new Definition('BazClass'));
  294. $config->setAlias('alias_for_foo', 'foo');
  295. $container->merge($config);
  296. $this->assertEquals(array('foo', 'bar', 'baz'), array_keys($container->getDefinitions()), '->merge() merges definitions already defined ones');
  297. $this->assertEquals(array('alias_for_foo' => 'foo'), $container->getAliases(), '->merge() registers defined aliases');
  298. $container = new ContainerBuilder();
  299. $container->register('foo', 'FooClass');
  300. $config->setDefinition('foo', new Definition('BazClass'));
  301. $container->merge($config);
  302. $this->assertEquals('BazClass', $container->getDefinition('foo')->getClass(), '->merge() overrides already defined services');
  303. }
  304. /**
  305. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::findTaggedServiceIds
  306. */
  307. public function testfindTaggedServiceIds()
  308. {
  309. $builder = new ContainerBuilder();
  310. $builder
  311. ->register('foo', 'FooClass')
  312. ->addTag('foo', array('foo' => 'foo'))
  313. ->addTag('bar', array('bar' => 'bar'))
  314. ->addTag('foo', array('foofoo' => 'foofoo'))
  315. ;
  316. $this->assertEquals($builder->findTaggedServiceIds('foo'), array(
  317. 'foo' => array(
  318. array('foo' => 'foo'),
  319. array('foofoo' => 'foofoo'),
  320. )
  321. ), '->findTaggedServiceIds() returns an array of service ids and its tag attributes');
  322. $this->assertEquals(array(), $builder->findTaggedServiceIds('foobar'), '->findTaggedServiceIds() returns an empty array if there is annotated services');
  323. }
  324. /**
  325. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::findDefinition
  326. */
  327. public function testFindDefinition()
  328. {
  329. $container = new ContainerBuilder();
  330. $container->setDefinition('foo', $definition = new Definition('FooClass'));
  331. $container->setAlias('bar', 'foo');
  332. $container->setAlias('foobar', 'bar');
  333. $this->assertEquals($definition, $container->findDefinition('foobar'), '->findDefinition() returns a Definition');
  334. }
  335. /**
  336. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::getResources
  337. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::addResource
  338. */
  339. public function testResources()
  340. {
  341. $container = new ContainerBuilder();
  342. $container->addResource($a = new FileResource(__DIR__.'/Fixtures/xml/services1.xml'));
  343. $container->addResource($b = new FileResource(__DIR__.'/Fixtures/xml/services2.xml'));
  344. $this->assertEquals(array($a, $b), $container->getResources(), '->getResources() returns an array of resources read for the current configuration');
  345. }
  346. /**
  347. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::registerExtension
  348. * @covers Symfony\Component\DependencyInjection\ContainerBuilder::getExtension
  349. */
  350. public function testExtension()
  351. {
  352. $container = new ContainerBuilder();
  353. $container->registerExtension($extension = new \ProjectExtension());
  354. $this->assertTrue($container->getExtension('project') === $extension, '->registerExtension() registers an extension');
  355. }
  356. }