View source
<?php
namespace Drupal\Tests\Component\DependencyInjection\Dumper;
use Drupal\Component\Utility\Crypt;
use Drupal\Tests\PhpUnitCompatibilityTrait;
use PHPUnit\Framework\TestCase;
use Symfony\Component\DependencyInjection\Definition;
use Symfony\Component\DependencyInjection\Reference;
use Symfony\Component\DependencyInjection\Parameter;
use Symfony\Component\ExpressionLanguage\Expression;
use Symfony\Component\DependencyInjection\ParameterBag\ParameterBag;
use Symfony\Component\DependencyInjection\ContainerInterface;
use Symfony\Component\DependencyInjection\Exception\InvalidArgumentException;
use Symfony\Component\DependencyInjection\Exception\RuntimeException;
class OptimizedPhpArrayDumperTest extends TestCase {
use PhpUnitCompatibilityTrait;
protected $containerBuilder;
protected $containerDefinition;
protected $machineFormat = TRUE;
protected $dumperClass = '\\Drupal\\Component\\DependencyInjection\\Dumper\\OptimizedPhpArrayDumper';
protected $dumper;
protected function setUp() : void {
$this->containerBuilder = $this
->prophesize('\\Symfony\\Component\\DependencyInjection\\ContainerBuilder');
$this->containerBuilder
->getAliases()
->willReturn([]);
$this->containerBuilder
->getParameterBag()
->willReturn(new ParameterBag());
$this->containerBuilder
->getDefinitions()
->willReturn([]);
$this->containerBuilder
->isCompiled()
->willReturn(TRUE);
$definition = [];
$definition['aliases'] = [];
$definition['parameters'] = [];
$definition['services'] = [];
$definition['frozen'] = TRUE;
$definition['machine_format'] = $this->machineFormat;
$this->containerDefinition = $definition;
$this->dumper = new $this->dumperClass($this->containerBuilder
->reveal());
}
public function testDumpForEmptyContainer() {
$serialized_definition = $this->dumper
->dump();
$this
->assertEquals(serialize($this->containerDefinition), $serialized_definition);
}
public function testGetAliases($aliases, $definition_aliases) {
$this->containerDefinition['aliases'] = $definition_aliases;
$this->containerBuilder
->getAliases()
->willReturn($aliases);
$this
->assertEquals($this->containerDefinition, $this->dumper
->getArray(), 'Expected definition matches dump.');
}
public function getAliasesDataProvider() {
return [
[
[],
[],
],
[
[
'foo' => 'foo.alias',
],
[
'foo' => 'foo.alias',
],
],
[
[
'foo' => 'foo.alias',
'foo.alias' => 'foo.alias.alias',
],
[
'foo' => 'foo.alias.alias',
'foo.alias' => 'foo.alias.alias',
],
],
];
}
public function testGetParameters($parameters, $definition_parameters, $is_frozen) {
$this->containerDefinition['parameters'] = $definition_parameters;
$this->containerDefinition['frozen'] = $is_frozen;
$parameter_bag = new ParameterBag($parameters);
$this->containerBuilder
->getParameterBag()
->willReturn($parameter_bag);
$this->containerBuilder
->isCompiled()
->willReturn($is_frozen);
$this
->assertEquals($this->containerDefinition, $this->dumper
->getArray(), 'Expected definition matches dump.');
}
public function getParametersDataProvider() {
return [
[
[],
[],
TRUE,
],
[
[
'foo' => 'value_foo',
],
[
'foo' => 'value_foo',
],
TRUE,
],
[
[
'foo' => [
'llama' => 'yes',
],
],
[
'foo' => [
'llama' => 'yes',
],
],
TRUE,
],
[
[
'foo' => '%llama%',
'llama' => 'yes',
],
[
'foo' => '%%llama%%',
'llama' => 'yes',
],
TRUE,
],
[
[
'foo' => '%llama%',
'llama' => 'yes',
],
[
'foo' => '%llama%',
'llama' => 'yes',
],
FALSE,
],
];
}
public function testGetServiceDefinitions($services, $definition_services) {
$this->containerDefinition['services'] = $definition_services;
$this->containerBuilder
->getDefinitions()
->willReturn($services);
$bar_definition = new Definition('\\stdClass');
$bar_definition
->setPublic(TRUE);
$this->containerBuilder
->getDefinition('bar')
->willReturn($bar_definition);
$private_definition = new Definition('\\stdClass');
$private_definition
->setPublic(FALSE);
$this->containerBuilder
->getDefinition('private_definition')
->willReturn($private_definition);
$this
->assertEquals($this->containerDefinition, $this->dumper
->getArray(), 'Expected definition matches dump.');
}
public function getDefinitionsDataProvider() {
$base_service_definition = [
'class' => '\\stdClass',
'public' => TRUE,
'file' => FALSE,
'synthetic' => FALSE,
'lazy' => FALSE,
'arguments' => [],
'arguments_count' => 0,
'properties' => [],
'calls' => [],
'shared' => TRUE,
'factory' => FALSE,
'configurator' => FALSE,
];
$service_definitions[] = [] + $base_service_definition;
$service_definitions[] = [
'public' => FALSE,
] + $base_service_definition;
$service_definitions[] = [
'file' => 'test_include.php',
] + $base_service_definition;
$service_definitions[] = [
'synthetic' => TRUE,
] + $base_service_definition;
$service_definitions[] = [
'shared' => FALSE,
] + $base_service_definition;
$service_definitions[] = [
'lazy' => TRUE,
] + $base_service_definition;
$service_definitions[] = [
'arguments' => [
'foo',
new Reference('bar'),
],
'arguments_count' => 2,
'arguments_expected' => $this
->getCollection([
'foo',
$this
->getServiceCall('bar'),
]),
] + $base_service_definition;
$reference = new Reference('bar', ContainerInterface::NULL_ON_INVALID_REFERENCE);
$service_definitions[] = [
'arguments' => [
$reference,
],
'arguments_count' => 1,
'arguments_expected' => $this
->getCollection([
$this
->getServiceCall('bar', ContainerInterface::NULL_ON_INVALID_REFERENCE),
]),
] + $base_service_definition;
$private_definition = [
'class' => '\\stdClass',
'public' => FALSE,
'arguments_count' => 0,
];
$service_definitions[] = [
'arguments' => [
'foo',
new Reference('private_definition'),
],
'arguments_count' => 2,
'arguments_expected' => $this
->getCollection([
'foo',
$this
->getPrivateServiceCall('private_definition', $private_definition, TRUE),
]),
] + $base_service_definition;
$private_definition_object = new Definition('\\stdClass');
$private_definition_object
->setPublic(FALSE);
$service_definitions[] = [
'arguments' => [
'foo',
$private_definition_object,
],
'arguments_count' => 2,
'arguments_expected' => $this
->getCollection([
'foo',
$this
->getPrivateServiceCall(NULL, $private_definition),
]),
] + $base_service_definition;
$service_definitions[] = [
'arguments' => [
[
[
'foo',
],
],
],
'arguments_count' => 1,
] + $base_service_definition;
$service_definitions[] = [
'arguments' => [
[
new Reference('bar'),
],
],
'arguments_count' => 1,
'arguments_expected' => $this
->getCollection([
$this
->getCollection([
$this
->getServiceCall('bar'),
]),
]),
] + $base_service_definition;
$service_definitions[] = [
'arguments' => [
new Parameter('llama_parameter'),
],
'arguments_count' => 1,
'arguments_expected' => $this
->getCollection([
$this
->getParameterCall('llama_parameter'),
]),
] + $base_service_definition;
$drupal_service = new \stdClass();
$drupal_service->_serviceId = 'bar';
$service_definitions[] = [
'arguments' => [
$drupal_service,
],
'arguments_count' => 1,
'arguments_expected' => $this
->getCollection([
$this
->getServiceCall('bar'),
]),
] + $base_service_definition;
$calls = [
[
'method',
$this
->getCollection([]),
],
[
'method2',
$this
->getCollection([]),
],
];
$service_definitions[] = [
'calls' => $calls,
] + $base_service_definition;
$service_definitions[] = [
'shared' => FALSE,
] + $base_service_definition;
$service_definitions[] = [
'factory' => [
new Reference('bar'),
'factoryMethod',
],
'factory_expected' => [
$this
->getServiceCall('bar'),
'factoryMethod',
],
] + $base_service_definition;
$service_definitions[] = [
'factory' => [
[
'foo',
'llama',
],
'factoryMethod',
],
] + $base_service_definition;
$service_definitions[] = [
'properties' => [
'_value' => 'llama',
],
] + $base_service_definition;
$service_definitions[] = [
'configurator' => [
new Reference('bar'),
'configureService',
],
'configurator_expected' => [
$this
->getServiceCall('bar'),
'configureService',
],
] + $base_service_definition;
$services_provided = [];
$services_provided[] = [
[],
[],
];
foreach ($service_definitions as $service_definition) {
$definition = $this
->prophesize('\\Symfony\\Component\\DependencyInjection\\Definition');
$definition
->getClass()
->willReturn($service_definition['class']);
$definition
->isPublic()
->willReturn($service_definition['public']);
$definition
->getFile()
->willReturn($service_definition['file']);
$definition
->isSynthetic()
->willReturn($service_definition['synthetic']);
$definition
->isLazy()
->willReturn($service_definition['lazy']);
$definition
->getArguments()
->willReturn($service_definition['arguments']);
$definition
->getProperties()
->willReturn($service_definition['properties']);
$definition
->getMethodCalls()
->willReturn($service_definition['calls']);
$definition
->isShared()
->willReturn($service_definition['shared']);
$definition
->getDecoratedService()
->willReturn(NULL);
$definition
->getFactory()
->willReturn($service_definition['factory']);
$definition
->getConfigurator()
->willReturn($service_definition['configurator']);
$filtered_service_definition = [];
foreach ($base_service_definition as $key => $value) {
$filtered_service_definition[$key] = $service_definition[$key];
unset($service_definition[$key]);
if ($key == 'class' || $key == 'arguments_count') {
continue;
}
if ($filtered_service_definition[$key] === $base_service_definition[$key]) {
unset($filtered_service_definition[$key]);
}
}
$filtered_service_definition += $service_definition;
foreach ([
'arguments',
'factory',
'configurator',
] as $key) {
$expected = $key . '_expected';
if (isset($filtered_service_definition[$expected])) {
$filtered_service_definition[$key] = $filtered_service_definition[$expected];
unset($filtered_service_definition[$expected]);
}
}
if (isset($filtered_service_definition['public']) && $filtered_service_definition['public'] === FALSE) {
$services_provided[] = [
[
'foo_service' => $definition
->reveal(),
],
[],
];
continue;
}
$services_provided[] = [
[
'foo_service' => $definition
->reveal(),
],
[
'foo_service' => $this
->serializeDefinition($filtered_service_definition),
],
];
}
return $services_provided;
}
protected function serializeDefinition(array $service_definition) {
return serialize($service_definition);
}
protected function getServiceCall($id, $invalid_behavior = ContainerInterface::EXCEPTION_ON_INVALID_REFERENCE) {
return (object) [
'type' => 'service',
'id' => $id,
'invalidBehavior' => $invalid_behavior,
];
}
public function testGetServiceDefinitionWithReferenceToAlias($public) {
$bar_definition = new Definition('\\stdClass');
$bar_definition_php_array = [
'class' => '\\stdClass',
];
if ($public) {
$bar_definition
->setPublic(TRUE);
}
else {
$bar_definition
->setPublic(FALSE);
$bar_definition_php_array['public'] = FALSE;
}
$bar_definition_php_array['arguments_count'] = 0;
$services['bar'] = $bar_definition;
$aliases['bar.alias'] = 'bar';
$foo = new Definition('\\stdClass');
$foo
->setPublic(TRUE);
$foo
->addArgument(new Reference('bar.alias'));
$services['foo'] = $foo;
$this->containerBuilder
->getAliases()
->willReturn($aliases);
$this->containerBuilder
->getDefinitions()
->willReturn($services);
$this->containerBuilder
->getDefinition('bar')
->willReturn($bar_definition);
$dump = $this->dumper
->getArray();
if ($public) {
$service_definition = $this
->getServiceCall('bar');
}
else {
$service_definition = $this
->getPrivateServiceCall('bar', $bar_definition_php_array, TRUE);
}
$data = [
'class' => '\\stdClass',
'arguments' => $this
->getCollection([
$service_definition,
]),
'arguments_count' => 1,
];
$this
->assertEquals($this
->serializeDefinition($data), $dump['services']['foo'], 'Expected definition matches dump.');
}
public function publicPrivateDataProvider() {
return [
[
TRUE,
],
[
FALSE,
],
];
}
public function testGetServiceDefinitionForDecoratedService() {
$bar_definition = new Definition('\\stdClass');
$bar_definition
->setPublic(TRUE);
$bar_definition
->setDecoratedService(new Reference('foo'));
$services['bar'] = $bar_definition;
$this->containerBuilder
->getDefinitions()
->willReturn($services);
$this
->expectException(InvalidArgumentException::class);
$this->dumper
->getArray();
}
public function testGetServiceDefinitionForExpression() {
$expression = new Expression('');
$bar_definition = new Definition('\\stdClass');
$bar_definition
->setPublic(TRUE);
$bar_definition
->addArgument($expression);
$services['bar'] = $bar_definition;
$this->containerBuilder
->getDefinitions()
->willReturn($services);
$this
->expectException(RuntimeException::class);
$this->dumper
->getArray();
}
public function testGetServiceDefinitionForObject() {
$service = new \stdClass();
$bar_definition = new Definition('\\stdClass');
$bar_definition
->setPublic(TRUE);
$bar_definition
->addArgument($service);
$services['bar'] = $bar_definition;
$this->containerBuilder
->getDefinitions()
->willReturn($services);
$this
->expectException(RuntimeException::class);
$this->dumper
->getArray();
}
public function testGetServiceDefinitionForResource() {
$resource = fopen('php://memory', 'r');
$bar_definition = new Definition('\\stdClass');
$bar_definition
->setPublic(TRUE);
$bar_definition
->addArgument($resource);
$services['bar'] = $bar_definition;
$this->containerBuilder
->getDefinitions()
->willReturn($services);
$this
->expectException(RuntimeException::class);
$this->dumper
->getArray();
}
public function testPercentsEscape($expected, $argument) {
$definition = new Definition('\\stdClass', [
$argument,
]);
$definition
->setPublic(TRUE);
$this->containerBuilder
->getDefinitions()
->willReturn([
'test' => $definition,
]);
$dump = $this->dumper
->getArray();
$this
->assertEquals($this
->serializeDefinition([
'class' => '\\stdClass',
'arguments' => $this
->getCollection([
$this
->getRaw($expected),
]),
'arguments_count' => 1,
]), $dump['services']['test']);
}
public function percentsEscapeProvider() {
return [
[
'%foo%',
'%%foo%%',
],
[
'foo%bar%',
'foo%%bar%%',
],
[
'%foo%bar',
'%%foo%%bar',
],
[
'%',
'%',
],
[
'%',
'%%',
],
[
'%%',
'%%%',
],
[
'%%',
'%%%%',
],
];
}
protected function getPrivateServiceCall($id, $service_definition, $shared = FALSE) {
if (!$id) {
$hash = Crypt::hashBase64(serialize($service_definition));
$id = 'private__' . $hash;
}
return (object) [
'type' => 'private_service',
'id' => $id,
'value' => $service_definition,
'shared' => $shared,
];
}
protected function getCollection($collection, $resolve = TRUE) {
return (object) [
'type' => 'collection',
'value' => $collection,
'resolve' => $resolve,
];
}
protected function getParameterCall($name) {
return (object) [
'type' => 'parameter',
'name' => $name,
];
}
protected function getRaw($value) {
return (object) [
'type' => 'raw',
'value' => $value,
];
}
}
namespace Symfony\Component\ExpressionLanguage;
if (!class_exists('\\Symfony\\Component\\ExpressionLanguage\\Expression')) {
class Expression {
public function __construct($expression) {
}
public function __toString() {
return 'dummy_expression';
}
}
}