123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196 |
- <?php
- declare(strict_types=1);
- /**
- * This file is part of Hyperf.
- *
- * @link https://www.hyperf.io
- * @document https://hyperf.wiki
- * @contact group@hyperf.io
- * @license https://github.com/hyperf/hyperf/blob/master/LICENSE
- */
- namespace Hyperf\Di;
- use Hyperf\Contract\ContainerInterface as HyperfContainerInterface;
- use Hyperf\Di\Definition\DefinitionInterface;
- use Hyperf\Di\Definition\ObjectDefinition;
- use Hyperf\Di\Exception\InvalidArgumentException;
- use Hyperf\Di\Exception\NotFoundException;
- use Hyperf\Di\Resolver\ResolverDispatcher;
- use Psr\Container\ContainerInterface as PsrContainerInterface;
- class Container implements HyperfContainerInterface
- {
- /**
- * Map of entries that are already resolved.
- *
- * @var array
- */
- private $resolvedEntries = [];
- /**
- * Map of definitions that are already fetched (local cache).
- */
- private $fetchedDefinitions = [];
- /**
- * @var Definition\DefinitionSourceInterface
- */
- private $definitionSource;
- /**
- * @var Resolver\ResolverInterface
- */
- private $definitionResolver;
- /**
- * Container constructor.
- */
- public function __construct(Definition\DefinitionSourceInterface $definitionSource)
- {
- $this->definitionSource = $definitionSource;
- $this->definitionResolver = new ResolverDispatcher($this);
- // Auto-register the container.
- $this->resolvedEntries = [
- self::class => $this,
- PsrContainerInterface::class => $this,
- HyperfContainerInterface::class => $this,
- ];
- }
- /**
- * Build an entry of the container by its name.
- * This method behave like get() except resolves the entry again every time.
- * For example if the entry is a class then a new instance will be created each time.
- * This method makes the container behave like a factory.
- *
- * @param string $name entry name or a class name
- * @param array $parameters Optional parameters to use to build the entry. Use this to force specific parameters
- * to specific values. Parameters not defined in this array will be resolved using
- * the container.
- * @throws NotFoundException no entry found for the given name
- * @throws InvalidArgumentException the name parameter must be of type string
- */
- public function make(string $name, array $parameters = [])
- {
- $definition = $this->getDefinition($name);
- if (! $definition) {
- throw new NotFoundException("No entry or class found for '{$name}'");
- }
- return $this->resolveDefinition($definition, $parameters);
- }
- /**
- * Bind an arbitrary resolved entry to an identifier.
- * Useful for testing 'get'.
- * @param mixed $entry
- */
- public function set(string $name, $entry)
- {
- $this->resolvedEntries[$name] = $entry;
- }
- /**
- * Unbind an arbitrary resolved entry.
- */
- public function unbind(string $name)
- {
- if ($this->has($name)) {
- unset($this->resolvedEntries[$name]);
- }
- }
- /**
- * Bind an arbitrary definition to an identifier.
- * Useful for testing 'make'.
- *
- * @param array|callable|string $definition
- */
- public function define(string $name, $definition)
- {
- $this->setDefinition($name, $definition);
- }
- /**
- * Finds an entry of the container by its identifier and returns it.
- *
- * @param string $name identifier of the entry to look for
- */
- public function get($name)
- {
- // If the entry is already resolved we return it
- if (isset($this->resolvedEntries[$name]) || array_key_exists($name, $this->resolvedEntries)) {
- return $this->resolvedEntries[$name];
- }
- return $this->resolvedEntries[$name] = $this->make($name);
- }
- /**
- * Returns true if the container can return an entry for the given identifier.
- * Returns false otherwise.
- * `has($name)` returning true does not mean that `get($name)` will not throw an exception.
- * It does however mean that `get($name)` will not throw a `NotFoundExceptionInterface`.
- *
- * @param mixed|string $name identifier of the entry to look for
- */
- public function has($name): bool
- {
- if (! is_string($name)) {
- throw new InvalidArgumentException(sprintf('The name parameter must be of type string, %s given', is_object($name) ? get_class($name) : gettype($name)));
- }
- if (array_key_exists($name, $this->resolvedEntries)) {
- return true;
- }
- $definition = $this->getDefinition($name);
- if ($definition === null) {
- return false;
- }
- if ($definition instanceof ObjectDefinition) {
- return $definition->isInstantiable();
- }
- return true;
- }
- public function getDefinitionSource(): Definition\DefinitionSourceInterface
- {
- return $this->definitionSource;
- }
- /**
- * @param array|callable|string $definition
- */
- private function setDefinition(string $name, $definition): void
- {
- // Clear existing entry if it exists
- if (array_key_exists($name, $this->resolvedEntries)) {
- unset($this->resolvedEntries[$name]);
- }
- $this->fetchedDefinitions = []; // Completely clear this local cache
- $this->definitionSource->addDefinition($name, $definition);
- }
- private function getDefinition(string $name): ?DefinitionInterface
- {
- // Local cache that avoids fetching the same definition twice
- if (! array_key_exists($name, $this->fetchedDefinitions)) {
- $this->fetchedDefinitions[$name] = $this->definitionSource->getDefinition($name);
- }
- return $this->fetchedDefinitions[$name];
- }
- /**
- * Resolves a definition.
- */
- private function resolveDefinition(DefinitionInterface $definition, array $parameters = [])
- {
- return $this->definitionResolver->resolve($definition, $parameters);
- }
- }
|