depcalc-8.x-1.x-dev/src/DependencyCalculator.php
src/DependencyCalculator.php
<?php
namespace Drupal\depcalc;
use Drupal\Core\Logger\LoggerChannelInterface;
use Drupal\depcalc\Event\CalculateEntityDependenciesEvent;
use Symfony\Component\EventDispatcher\EventDispatcherInterface;
/**
* Calculates all the dependencies of a given entity.
*
* This class calculates all the dependencies of any entity. Pass an entity of
* any sort to the calculateDependencies() method, and this class will recurse
* through all the existing inter-dependencies that it knows about.
*/
class DependencyCalculator {
/**
* The event dispatcher.
*
* @var \Symfony\Component\EventDispatcher\EventDispatcherInterface
*/
protected $dispatcher;
/**
* The depcalc logger channel.
*
* @var \Drupal\Core\Logger\LoggerChannelInterface
*/
protected $logger;
/**
* The DependencyCalculator constructor.
*
* @param \Symfony\Component\EventDispatcher\EventDispatcherInterface $dispatcher
* The event dispatcher.
* @param \Drupal\Core\Logger\LoggerChannelInterface $logger
* The depcalc logger channel.
*/
public function __construct(EventDispatcherInterface $dispatcher, LoggerChannelInterface $logger) {
$this->dispatcher = $dispatcher;
$this->logger = $logger;
}
/**
* Calculates the dependencies.
*
* @param \Drupal\depcalc\DependentEntityWrapperInterface $wrapper
* The dependency wrapper for the entity to calculate dependencies.
* @param \Drupal\depcalc\DependencyStack $stack
* An array of pre-calculated dependencies to prevent recalculation.
* @param array $dependencies
* (optional) An array of dependencies by reference. Internally used.
*
* @return array
* The list of the dependencies.
*/
public function calculateDependencies(DependentEntityWrapperInterface $wrapper, DependencyStack $stack, array &$dependencies = []) {
// Prevent config dependency calculation if it is disabled.
if ($stack->shouldIgnoreConfig() && $wrapper->isConfigEntity()) {
return $dependencies;
}
if (empty($dependencies['module'])) {
$dependencies['module'] = [];
}
// Prevent handling the same entity more than once.
if (!empty($dependencies[$wrapper->getUuid()])) {
return $dependencies;
}
// Prevent handling the same entity more than once.
if ($dependency = $stack->getDependency($wrapper->getUuid())) {
// Get module dependencies from cache.
$modules = $dependency->getModuleDependencies();
if ($modules) {
$wrapper->addModuleDependencies($modules);
$dependencies['module'] = $modules;
}
// Tries to get a list of the dependencies from cache.
try {
$dependencies = $stack->getDependenciesByUuid(array_keys($dependency->getDependencies()));
// Add the dependencies from cache to the main wrapper as well as stack.
$wrapper->addDependencies($stack, ...array_values($dependencies));
$dependencies[$wrapper->getUuid()] = $dependency;
return $dependencies;
}
catch (\Exception $exception) {
$msg = $exception->getMessage();
$missing_dep_msg = sprintf("Retrieving dependencies from cache failed for entity (%s, %s) as one of the dependency is missing from the cache. Dependencies will be re-calculated." . PHP_EOL, $wrapper->getEntityTypeId(), $wrapper->getUuid());
$this->logger->warning($missing_dep_msg . $msg);
}
}
// We add the dependency to the stack because we
// need it there but the dependency data is WRONG,
// so don't make it permanent in case something breaks before we get to
// resave it later when all dependencies are correctly calculated.
$stack->addDependency($wrapper, FALSE);
$event = new CalculateEntityDependenciesEvent($wrapper, $stack);
$this->dispatcher->dispatch($event, DependencyCalculatorEvents::CALCULATE_DEPENDENCIES);
// Update the stack with the newest $wrapper and the correct dependencies.
$stack->addDependency($wrapper);
$modules = $event->getModuleDependencies();
if ($modules) {
$wrapper->addModuleDependencies($modules);
}
$dependencies = $stack->getDependenciesByUuid(array_keys($event->getDependencies()));
$wrapper->addDependencies($stack, ...array_values($dependencies));
$dependencies[$wrapper->getUuid()] = $wrapper;
// Extract the name of the module providing this entity type.
$entity_module = $wrapper->getEntity()->getEntityType()->getProvider();
$entity_module = \Drupal::moduleHandler()->moduleExists($entity_module) ? [$entity_module] : [];
$dependencies['module'] = $event->getModuleDependencies() + $entity_module;
return $dependencies;
}
}
