Skip to content

Commit cc4c218

Browse files
authored
Merge pull request #43 from GeeH/geeh/#3-hotfix-from-old-repo
#3 Ported 279 from ZF repository - fixes merging of duplicate delegator factory definitions
2 parents 7c42b69 + 316fa33 commit cc4c218

File tree

2 files changed

+77
-3
lines changed

2 files changed

+77
-3
lines changed

src/ServiceManager.php

+29-3
Original file line numberDiff line numberDiff line change
@@ -21,9 +21,9 @@
2121
use ProxyManager\GeneratorStrategy\EvaluatingGeneratorStrategy;
2222
use ProxyManager\GeneratorStrategy\FileWriterGeneratorStrategy;
2323
use Psr\Container\ContainerInterface;
24+
use Laminas\Stdlib\ArrayUtils;
2425

2526
use function array_intersect;
26-
use function array_merge_recursive;
2727
use function class_exists;
2828
use function get_class;
2929
use function gettype;
@@ -363,7 +363,7 @@ public function configure(array $config)
363363
}
364364

365365
if (isset($config['delegators'])) {
366-
$this->delegators = array_merge_recursive($this->delegators, $config['delegators']);
366+
$this->mergeDelegators($config['delegators']);
367367
}
368368

369369
if (isset($config['shared'])) {
@@ -384,7 +384,7 @@ public function configure(array $config)
384384
// If lazy service configuration was provided, reset the lazy services
385385
// delegator factory.
386386
if (isset($config['lazy_services']) && ! empty($config['lazy_services'])) {
387-
$this->lazyServices = array_merge_recursive($this->lazyServices, $config['lazy_services']);
387+
$this->lazyServices = ArrayUtils::merge($this->lazyServices, $config['lazy_services']);
388388
$this->lazyServicesDelegator = null;
389389
}
390390

@@ -737,6 +737,32 @@ private function createLazyServiceDelegatorFactory()
737737
return $this->lazyServicesDelegator;
738738
}
739739

740+
/**
741+
* Merge delegators avoiding multiple same delegators for the same service.
742+
* It works with strings and class instances.
743+
* It's not possible to de-duple anonymous functions
744+
*
745+
* @param string[][]|Factory\DelegatorFactoryInterface[][] $config
746+
* @return string[][]|Factory\DelegatorFactoryInterface[][]
747+
*/
748+
private function mergeDelegators(array $config)
749+
{
750+
foreach ($config as $key => $delegators) {
751+
if (! array_key_exists($key, $this->delegators)) {
752+
$this->delegators[$key] = $delegators;
753+
continue;
754+
}
755+
756+
foreach ($delegators as $delegator) {
757+
if (! in_array($delegator, $this->delegators[$key], true)) {
758+
$this->delegators[$key][] = $delegator;
759+
}
760+
}
761+
}
762+
763+
return $this->delegators;
764+
}
765+
740766
/**
741767
* Create aliases and factories for invokable classes.
742768
*

test/ServiceManagerTest.php

+48
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@
1515
use Laminas\ServiceManager\ServiceManager;
1616
use LaminasTest\ServiceManager\TestAsset\InvokableObject;
1717
use LaminasTest\ServiceManager\TestAsset\SimpleServiceManager;
18+
use Laminas\ServiceManager\Proxy\LazyServiceFactory;
1819
use PHPUnit\Framework\TestCase;
1920
use Psr\Container\ContainerInterface;
2021
use stdClass;
@@ -364,4 +365,51 @@ public function testResolvedAliasNoMatchingAbstractFactoryReturnsFalse()
364365

365366
self::assertFalse($serviceManager->has('Alias'));
366367
}
368+
369+
/**
370+
* @group #3
371+
* @see https://github.com/laminas/laminas-servicemanager/issues/3
372+
*/
373+
public function testConfiguringADelegatorMultipleTimesDoesNotLeadToDuplicateDelegatorCalls()
374+
{
375+
$delegatorFactory = function (
376+
ContainerInterface $container,
377+
$name,
378+
callable $callback
379+
) {
380+
/** @var InvokableObject $instance */
381+
$instance = $callback();
382+
$options = $instance->getOptions();
383+
$inc = $options['inc'] ?? 0;
384+
return new InvokableObject(['inc' => ++$inc]);
385+
};
386+
387+
$config = [
388+
'factories' => [
389+
'Foo' => function () {
390+
return new InvokableObject();
391+
},
392+
],
393+
'delegators' => [
394+
'Foo' => [
395+
$delegatorFactory,
396+
LazyServiceFactory::class,
397+
],
398+
],
399+
'lazy_services' => [
400+
'class_map' => [
401+
'Foo' => InvokableObject::class,
402+
],
403+
],
404+
];
405+
406+
$serviceManager = new ServiceManager($config);
407+
$serviceManager->configure($config);
408+
409+
/** @var InvokableObject $instance */
410+
$instance = $serviceManager->get('Foo');
411+
412+
self::assertInstanceOf(InvokableObject::class, $instance);
413+
self::assertSame(1, $instance->getOptions()['inc']);
414+
}
367415
}

0 commit comments

Comments
 (0)