class ReverseContainer
Retrieves service IDs from the container for public services.
Heavily inspired by \Symfony\Component\DependencyInjection\ReverseContainer.
Hierarchy
- class \Drupal\Component\DependencyInjection\ReverseContainer
Expanded class hierarchy of ReverseContainer
6 files declare their use of ReverseContainer
- DecoratedServiceTest.php in core/modules/ system/ tests/ src/ Kernel/ DecoratedServiceTest.php 
- DependencySerializationTest.php in core/tests/ Drupal/ Tests/ Core/ DependencyInjection/ DependencySerializationTest.php 
- DependencySerializationTrait.php in core/lib/ Drupal/ Core/ DependencyInjection/ DependencySerializationTrait.php 
- DrupalKernel.php in core/lib/ Drupal/ Core/ DrupalKernel.php 
- ReverseContainerTest.php in core/tests/ Drupal/ Tests/ Component/ DependencyInjection/ ReverseContainerTest.php 
File
- 
              core/lib/ Drupal/ Component/ DependencyInjection/ ReverseContainer.php, line 13 
Namespace
Drupal\Component\DependencyInjectionView source
final class ReverseContainer {
  
  /**
   * A closure on the container that can search for services.
   *
   * @var \Closure
   */
  private \Closure $getServiceId;
  
  /**
   * A static map of services to a hash.
   *
   * @var array
   */
  private static array $recordedServices = [];
  
  /**
   * Constructs a ReverseContainer object.
   *
   * @param \Drupal\Component\DependencyInjection\Container|\Symfony\Component\DependencyInjection\Container $serviceContainer
   *   The service container.
   */
  public function __construct(private readonly Container|SymfonyContainer $serviceContainer) {
    $this->getServiceId = \Closure::bind(function ($service) : ?string {
      return array_search($service, $this->services, TRUE) ?: NULL;
    }, $serviceContainer, $serviceContainer);
  }
  
  /**
   * Returns the ID of the passed object when it exists as a service.
   *
   * To be reversible, services need to be public.
   *
   * @param object $service
   *   The service to find the ID for.
   */
  public function getId(object $service) : ?string {
    if ($this->serviceContainer === $service || $service instanceof SymfonyContainerInterface) {
      return 'service_container';
    }
    $hash = $this->generateServiceIdHash($service);
    $id = self::$recordedServices[$hash] ?? ($this->getServiceId)($service);
    if ($id !== NULL && $this->serviceContainer
      ->has($id)) {
      self::$recordedServices[$hash] = $id;
      return $id;
    }
    return NULL;
  }
  
  /**
   * Records a map of the container's services.
   *
   * This method is used so that stale services can be serialized after a
   * container has been re-initialized.
   */
  public function recordContainer() : void {
    $service_recorder = \Closure::bind(function () : array {
      return $this->services;
    }, $this->serviceContainer, $this->serviceContainer);
    self::$recordedServices = array_merge(self::$recordedServices, array_flip(array_map([
      $this,
      'generateServiceIdHash',
    ], $service_recorder())));
  }
  
  /**
   * Generates an identifier for a service based on the object class and hash.
   *
   * @param object $object
   *   The object to generate an identifier for.
   *
   * @return string
   *   The object's class and hash concatenated together.
   */
  private function generateServiceIdHash(object $object) : string {
    // Include class name as an additional namespace for the hash since
    // spl_object_hash's return can be recycled. This still is not a 100%
    // guarantee to be unique but makes collisions incredibly difficult and even
    // then the interface would be preserved.
    // @see https://php.net/spl_object_hash#refsect1-function.spl-object-hash-notes
    return get_class($object) . spl_object_hash($object);
  }
}Members
| Title Sort descending | Modifiers | Object type | Summary | 
|---|---|---|---|
| ReverseContainer::$getServiceId | private | property | A closure on the container that can search for services. | 
| ReverseContainer::$recordedServices | private static | property | A static map of services to a hash. | 
| ReverseContainer::generateServiceIdHash | private | function | Generates an identifier for a service based on the object class and hash. | 
| ReverseContainer::getId | public | function | Returns the ID of the passed object when it exists as a service. | 
| ReverseContainer::recordContainer | public | function | Records a map of the container's services. | 
| ReverseContainer::__construct | public | function | Constructs a ReverseContainer object. | 
Buggy or inaccurate documentation? Please file an issue. Need support? Need help programming? Connect with the Drupal community.
