249 lines
6.6 KiB
PHP
249 lines
6.6 KiB
PHP
|
<?php declare(strict_types=1);
|
||
|
|
||
|
namespace Automattic\WooCommerce\Vendor\League\Container;
|
||
|
|
||
|
use Automattic\WooCommerce\Vendor\League\Container\Definition\{DefinitionAggregate, DefinitionInterface, DefinitionAggregateInterface};
|
||
|
use Automattic\WooCommerce\Vendor\League\Container\Exception\{NotFoundException, ContainerException};
|
||
|
use Automattic\WooCommerce\Vendor\League\Container\Inflector\{InflectorAggregate, InflectorInterface, InflectorAggregateInterface};
|
||
|
use Automattic\WooCommerce\Vendor\League\Container\ServiceProvider\{
|
||
|
ServiceProviderAggregate,
|
||
|
ServiceProviderAggregateInterface,
|
||
|
ServiceProviderInterface
|
||
|
};
|
||
|
use Psr\Container\ContainerInterface;
|
||
|
|
||
|
class Container implements ContainerInterface
|
||
|
{
|
||
|
/**
|
||
|
* @var boolean
|
||
|
*/
|
||
|
protected $defaultToShared = false;
|
||
|
|
||
|
/**
|
||
|
* @var DefinitionAggregateInterface
|
||
|
*/
|
||
|
protected $definitions;
|
||
|
|
||
|
/**
|
||
|
* @var ServiceProviderAggregateInterface
|
||
|
*/
|
||
|
protected $providers;
|
||
|
|
||
|
/**
|
||
|
* @var InflectorAggregateInterface
|
||
|
*/
|
||
|
protected $inflectors;
|
||
|
|
||
|
/**
|
||
|
* @var ContainerInterface[]
|
||
|
*/
|
||
|
protected $delegates = [];
|
||
|
|
||
|
/**
|
||
|
* Construct.
|
||
|
*
|
||
|
* @param DefinitionAggregateInterface|null $definitions
|
||
|
* @param ServiceProviderAggregateInterface|null $providers
|
||
|
* @param InflectorAggregateInterface|null $inflectors
|
||
|
*/
|
||
|
public function __construct(
|
||
|
DefinitionAggregateInterface $definitions = null,
|
||
|
ServiceProviderAggregateInterface $providers = null,
|
||
|
InflectorAggregateInterface $inflectors = null
|
||
|
) {
|
||
|
$this->definitions = $definitions ?? new DefinitionAggregate;
|
||
|
$this->providers = $providers ?? new ServiceProviderAggregate;
|
||
|
$this->inflectors = $inflectors ?? new InflectorAggregate;
|
||
|
|
||
|
if ($this->definitions instanceof ContainerAwareInterface) {
|
||
|
$this->definitions->setLeagueContainer($this);
|
||
|
}
|
||
|
|
||
|
if ($this->providers instanceof ContainerAwareInterface) {
|
||
|
$this->providers->setLeagueContainer($this);
|
||
|
}
|
||
|
|
||
|
if ($this->inflectors instanceof ContainerAwareInterface) {
|
||
|
$this->inflectors->setLeagueContainer($this);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Add an item to the container.
|
||
|
*
|
||
|
* @param string $id
|
||
|
* @param mixed $concrete
|
||
|
* @param boolean $shared
|
||
|
*
|
||
|
* @return DefinitionInterface
|
||
|
*/
|
||
|
public function add(string $id, $concrete = null, bool $shared = null) : DefinitionInterface
|
||
|
{
|
||
|
$concrete = $concrete ?? $id;
|
||
|
$shared = $shared ?? $this->defaultToShared;
|
||
|
|
||
|
return $this->definitions->add($id, $concrete, $shared);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Proxy to add with shared as true.
|
||
|
*
|
||
|
* @param string $id
|
||
|
* @param mixed $concrete
|
||
|
*
|
||
|
* @return DefinitionInterface
|
||
|
*/
|
||
|
public function share(string $id, $concrete = null) : DefinitionInterface
|
||
|
{
|
||
|
return $this->add($id, $concrete, true);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Whether the container should default to defining shared definitions.
|
||
|
*
|
||
|
* @param boolean $shared
|
||
|
*
|
||
|
* @return self
|
||
|
*/
|
||
|
public function defaultToShared(bool $shared = true) : ContainerInterface
|
||
|
{
|
||
|
$this->defaultToShared = $shared;
|
||
|
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get a definition to extend.
|
||
|
*
|
||
|
* @param string $id [description]
|
||
|
*
|
||
|
* @return DefinitionInterface
|
||
|
*/
|
||
|
public function extend(string $id) : DefinitionInterface
|
||
|
{
|
||
|
if ($this->providers->provides($id)) {
|
||
|
$this->providers->register($id);
|
||
|
}
|
||
|
|
||
|
if ($this->definitions->has($id)) {
|
||
|
return $this->definitions->getDefinition($id);
|
||
|
}
|
||
|
|
||
|
throw new NotFoundException(
|
||
|
sprintf('Unable to extend alias (%s) as it is not being managed as a definition', $id)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Add a service provider.
|
||
|
*
|
||
|
* @param ServiceProviderInterface|string $provider
|
||
|
*
|
||
|
* @return self
|
||
|
*/
|
||
|
public function addServiceProvider($provider) : self
|
||
|
{
|
||
|
$this->providers->add($provider);
|
||
|
|
||
|
return $this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* {@inheritdoc}
|
||
|
*/
|
||
|
public function get($id, bool $new = false)
|
||
|
{
|
||
|
if ($this->definitions->has($id)) {
|
||
|
$resolved = $this->definitions->resolve($id, $new);
|
||
|
return $this->inflectors->inflect($resolved);
|
||
|
}
|
||
|
|
||
|
if ($this->definitions->hasTag($id)) {
|
||
|
$arrayOf = $this->definitions->resolveTagged($id, $new);
|
||
|
|
||
|
array_walk($arrayOf, function (&$resolved) {
|
||
|
$resolved = $this->inflectors->inflect($resolved);
|
||
|
});
|
||
|
|
||
|
return $arrayOf;
|
||
|
}
|
||
|
|
||
|
if ($this->providers->provides($id)) {
|
||
|
$this->providers->register($id);
|
||
|
|
||
|
if (!$this->definitions->has($id) && !$this->definitions->hasTag($id)) {
|
||
|
throw new ContainerException(sprintf('Service provider lied about providing (%s) service', $id));
|
||
|
}
|
||
|
|
||
|
return $this->get($id, $new);
|
||
|
}
|
||
|
|
||
|
foreach ($this->delegates as $delegate) {
|
||
|
if ($delegate->has($id)) {
|
||
|
$resolved = $delegate->get($id);
|
||
|
return $this->inflectors->inflect($resolved);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
throw new NotFoundException(sprintf('Alias (%s) is not being managed by the container or delegates', $id));
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* {@inheritdoc}
|
||
|
*/
|
||
|
public function has($id) : bool
|
||
|
{
|
||
|
if ($this->definitions->has($id)) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
if ($this->definitions->hasTag($id)) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
if ($this->providers->provides($id)) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
foreach ($this->delegates as $delegate) {
|
||
|
if ($delegate->has($id)) {
|
||
|
return true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Allows for manipulation of specific types on resolution.
|
||
|
*
|
||
|
* @param string $type
|
||
|
* @param callable|null $callback
|
||
|
*
|
||
|
* @return InflectorInterface
|
||
|
*/
|
||
|
public function inflector(string $type, callable $callback = null) : InflectorInterface
|
||
|
{
|
||
|
return $this->inflectors->add($type, $callback);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Delegate a backup container to be checked for services if it
|
||
|
* cannot be resolved via this container.
|
||
|
*
|
||
|
* @param ContainerInterface $container
|
||
|
*
|
||
|
* @return self
|
||
|
*/
|
||
|
public function delegate(ContainerInterface $container) : self
|
||
|
{
|
||
|
$this->delegates[] = $container;
|
||
|
|
||
|
if ($container instanceof ContainerAwareInterface) {
|
||
|
$container->setLeagueContainer($this);
|
||
|
}
|
||
|
|
||
|
return $this;
|
||
|
}
|
||
|
}
|