vendor/contao/core-bundle/src/Resources/contao/library/Contao/Model/Registry.php line 108

Open in your IDE?
  1. <?php
  2. /*
  3.  * This file is part of Contao.
  4.  *
  5.  * (c) Leo Feyer
  6.  *
  7.  * @license LGPL-3.0-or-later
  8.  */
  9. namespace Contao\Model;
  10. use Contao\Model;
  11. /**
  12.  * Handle a set of models
  13.  *
  14.  * The class handles traversing a set of models and lazy loads the database
  15.  * result rows upon their first usage.
  16.  */
  17. class Registry implements \Countable
  18. {
  19.     /**
  20.      * Object instance (Singleton)
  21.      * @var static
  22.      */
  23.     protected static $objInstance;
  24.     /**
  25.      * Models by table and PK
  26.      * @var array
  27.      */
  28.     protected $arrRegistry = array();
  29.     /**
  30.      * Aliases to PK's by table and column
  31.      * @var array
  32.      */
  33.     protected $arrAliases = array();
  34.     /**
  35.      * Models by object hash
  36.      * @var array
  37.      */
  38.     protected $arrIdentities = array();
  39.     /**
  40.      * Prevent direct instantiation (Singleton)
  41.      */
  42.     protected function __construct()
  43.     {
  44.     }
  45.     /**
  46.      * Prevent cloning of the object (Singleton)
  47.      */
  48.     final public function __clone()
  49.     {
  50.     }
  51.     /**
  52.      * Return the current object instance (Singleton)
  53.      *
  54.      * @return static The object instance
  55.      */
  56.     public static function getInstance()
  57.     {
  58.         if (static::$objInstance === null)
  59.         {
  60.             static::$objInstance = new static();
  61.         }
  62.         return static::$objInstance;
  63.     }
  64.     /**
  65.      * Reset the registry
  66.      *
  67.      * @internal Do not use this method in your code; it should only be called during kernel.reset
  68.      */
  69.     public function reset()
  70.     {
  71.         $this->arrRegistry = array();
  72.         $this->arrAliases = array();
  73.         $this->arrIdentities = array();
  74.     }
  75.     /**
  76.      * Count the elements
  77.      *
  78.      * @return integer The number of models
  79.      */
  80.     #[\ReturnTypeWillChange]
  81.     public function count()
  82.     {
  83.         return \count($this->arrIdentities);
  84.     }
  85.     /**
  86.      * Fetch a model by table name and primary key
  87.      *
  88.      * @param string $strTable The table name
  89.      * @param mixed  $varKey   The key
  90.      * @param string $strAlias An optional alias
  91.      *
  92.      * @return Model|null The model or null
  93.      */
  94.     public function fetch($strTable$varKey$strAlias=null)
  95.     {
  96.         /** @var Model $strClass */
  97.         $strClass Model::getClassFromTable($strTable);
  98.         $strPk $strClass::getPk();
  99.         // Search by PK (most common case)
  100.         if ($strAlias === null || $strAlias == $strPk)
  101.         {
  102.             return $this->arrRegistry[$strTable][$varKey] ?? null;
  103.         }
  104.         // Try to find the model by one of its aliases
  105.         return $this->fetchByAlias($strTable$strAlias$varKey);
  106.     }
  107.     /**
  108.      * Fetch a model by one of its aliases
  109.      *
  110.      * @param string $strTable The table name
  111.      * @param string $strAlias The alias
  112.      * @param mixed  $varValue The alias value
  113.      *
  114.      * @return Model|null The model or null
  115.      */
  116.     public function fetchByAlias($strTable$strAlias$varValue)
  117.     {
  118.         if (isset($this->arrAliases[$strTable][$strAlias][$varValue]))
  119.         {
  120.             $strPk $this->arrAliases[$strTable][$strAlias][$varValue];
  121.             if (isset($this->arrRegistry[$strTable][$strPk]))
  122.             {
  123.                 return $this->arrRegistry[$strTable][$strPk];
  124.             }
  125.         }
  126.         return null;
  127.     }
  128.     /**
  129.      * Register a model in the registry
  130.      *
  131.      * @param Model $objModel The model object
  132.      *
  133.      * @throws \RuntimeException If the instance exists already
  134.      */
  135.     public function register(Model $objModel)
  136.     {
  137.         $intObjectId spl_object_id($objModel);
  138.         // The model has been registered already
  139.         if (isset($this->arrIdentities[$intObjectId]))
  140.         {
  141.             return;
  142.         }
  143.         $strTable $objModel->getTable();
  144.         if (!isset($this->arrAliases[$strTable]))
  145.         {
  146.             $this->arrAliases[$strTable] = array();
  147.         }
  148.         if (!isset($this->arrRegistry[$strTable]))
  149.         {
  150.             $this->arrRegistry[$strTable] = array();
  151.         }
  152.         $strPk $objModel->getPk();
  153.         $varPk $objModel->$strPk;
  154.         if ($varPk === null)
  155.         {
  156.             throw new \RuntimeException('The primary key has not been set');
  157.         }
  158.         // Another model object is pointing to the DB record already
  159.         if (isset($this->arrRegistry[$strTable][$varPk]))
  160.         {
  161.             throw new \RuntimeException("The registry already contains an instance for $strTable::$strPk($varPk)");
  162.         }
  163.         $this->arrIdentities[$intObjectId] = $objModel;
  164.         $this->arrRegistry[$strTable][$varPk] = $objModel;
  165.         // Allow the model to modify the registry
  166.         $objModel->onRegister($this);
  167.     }
  168.     /**
  169.      * Unregister a model from the registry
  170.      *
  171.      * @param Model $objModel The model object
  172.      */
  173.     public function unregister(Model $objModel)
  174.     {
  175.         $intObjectId spl_object_id($objModel);
  176.         // The model is not registered
  177.         if (!isset($this->arrIdentities[$intObjectId]))
  178.         {
  179.             return;
  180.         }
  181.         $strTable $objModel->getTable();
  182.         $strPk    $objModel->getPk();
  183.         $intPk    $objModel->$strPk;
  184.         unset($this->arrIdentities[$intObjectId], $this->arrRegistry[$strTable][$intPk]);
  185.         // Allow the model to modify the registry
  186.         $objModel->onUnregister($this);
  187.     }
  188.     /**
  189.      * Check if a model is registered
  190.      *
  191.      * @param Model $objModel The model object
  192.      *
  193.      * @return boolean True if the model is registered
  194.      */
  195.     public function isRegistered(Model $objModel)
  196.     {
  197.         $intObjectId spl_object_id($objModel);
  198.         return isset($this->arrIdentities[$intObjectId]);
  199.     }
  200.     /**
  201.      * Register an alias for a model
  202.      *
  203.      * @param Model  $objModel The model object
  204.      * @param string $strAlias The alias name
  205.      * @param mixed  $varValue The value of the alias
  206.      *
  207.      * @throws \RuntimeException If the alias is already registered
  208.      */
  209.     public function registerAlias(Model $objModel$strAlias$varValue)
  210.     {
  211.         $strTable $objModel->getTable();
  212.         $strPk    $objModel->getPk();
  213.         $varPk    $objModel->$strPk;
  214.         if (isset($this->arrAliases[$strTable][$strAlias][$varValue]))
  215.         {
  216.             throw new \RuntimeException("The registry already contains an alias for $strTable::$strPk($varPk) ($strAlias/$varValue)");
  217.         }
  218.         $this->arrAliases[$strTable][$strAlias][$varValue] = $varPk;
  219.     }
  220.     /**
  221.      * Unregister an alias
  222.      *
  223.      * @param Model  $objModel The model object
  224.      * @param string $strAlias The alias name
  225.      * @param mixed  $varValue The value of the alias
  226.      *
  227.      * @throws \InvalidArgumentException If the alias is not registered
  228.      */
  229.     public function unregisterAlias(Model $objModel$strAlias$varValue)
  230.     {
  231.         $strTable $objModel->getTable();
  232.         if (!isset($this->arrAliases[$strTable][$strAlias][$varValue]))
  233.         {
  234.             $strPk $objModel->getPk();
  235.             $varPk $objModel->$strPk;
  236.             throw new \RuntimeException("The registry does not contain an alias for $strTable::$strPk($varPk) ($strAlias/$varValue)");
  237.         }
  238.         unset($this->arrAliases[$strTable][$strAlias][$varValue]);
  239.     }
  240.     /**
  241.      * Check if an alias is registered
  242.      *
  243.      * @param Model  $objModel The model object
  244.      * @param string $strAlias The alias name
  245.      * @param mixed  $varValue The value of the alias
  246.      *
  247.      * @return boolean True if the alias is registered
  248.      */
  249.     public function isRegisteredAlias(Model $objModel$strAlias$varValue)
  250.     {
  251.         $strTable $objModel->getTable();
  252.         return isset($this->arrAliases[$strTable][$strAlias][$varValue]);
  253.     }
  254. }
  255. class_alias(Registry::class, 'Model\Registry');