You are here

class MatcherDumper in Drupal 9

Same name in this branch
  1. 9 core/lib/Drupal/Core/Routing/MatcherDumper.php \Drupal\Core\Routing\MatcherDumper
  2. 9 core/lib/Drupal/Core/ProxyClass/Routing/MatcherDumper.php \Drupal\Core\ProxyClass\Routing\MatcherDumper
Same name and namespace in other branches
  1. 8 core/lib/Drupal/Core/Routing/MatcherDumper.php \Drupal\Core\Routing\MatcherDumper
  2. 10 core/lib/Drupal/Core/Routing/MatcherDumper.php \Drupal\Core\Routing\MatcherDumper

Dumps Route information to a database table.

Hierarchy

Expanded class hierarchy of MatcherDumper

See also

\Drupal\Core\Routing\RouteProvider

2 files declare their use of MatcherDumper
MatcherDumperTest.php in core/tests/Drupal/KernelTests/Core/Routing/MatcherDumperTest.php
RouteProviderTest.php in core/tests/Drupal/KernelTests/Core/Routing/RouteProviderTest.php
Contains \Drupal\KernelTests\Core\Routing\RouteProviderTest.
1 string reference to 'MatcherDumper'
core.services.yml in core/core.services.yml
core/core.services.yml
1 service uses MatcherDumper
router.dumper in core/core.services.yml
Drupal\Core\Routing\MatcherDumper

File

core/lib/Drupal/Core/Routing/MatcherDumper.php, line 16

Namespace

Drupal\Core\Routing
View source
class MatcherDumper implements MatcherDumperInterface {

  /**
   * The database connection to which to dump route information.
   *
   * @var \Drupal\Core\Database\Connection
   */
  protected $connection;

  /**
   * The routes to be dumped.
   *
   * @var \Symfony\Component\Routing\RouteCollection
   */
  protected $routes;

  /**
   * The state.
   *
   * @var \Drupal\Core\State\StateInterface
   */
  protected $state;

  /**
   * The name of the SQL table to which to dump the routes.
   *
   * @var string
   */
  protected $tableName;

  /**
   * Construct the MatcherDumper.
   *
   * @param \Drupal\Core\Database\Connection $connection
   *   The database connection which will be used to store the route
   *   information.
   * @param \Drupal\Core\State\StateInterface $state
   *   The state.
   * @param string $table
   *   (optional) The table to store the route info in. Defaults to 'router'.
   */
  public function __construct(Connection $connection, StateInterface $state, $table = 'router') {
    $this->connection = $connection;
    $this->state = $state;
    $this->tableName = $table;
  }

  /**
   * {@inheritdoc}
   */
  public function addRoutes(RouteCollection $routes) {
    if (empty($this->routes)) {
      $this->routes = $routes;
    }
    else {
      $this->routes
        ->addCollection($routes);
    }
  }

  /**
   * Dumps a set of routes to the router table in the database.
   *
   * Available options:
   * - provider: The route grouping that is being dumped. All existing
   *   routes with this provider will be deleted on dump.
   * - base_class: The base class name.
   *
   * @param array $options
   *   An array of options.
   */
  public function dump(array $options = []) {

    // Convert all of the routes into database records.
    // Accumulate the menu masks on top of any we found before.
    $masks = array_flip($this->state
      ->get('routing.menu_masks.' . $this->tableName, []));

    // Delete any old records first, then insert the new ones. That avoids
    // stale data. The transaction makes it atomic to avoid unstable router
    // states due to random failures.
    $transaction = $this->connection
      ->startTransaction();
    try {

      // We don't use truncate, because it is not guaranteed to be transaction
      // safe.
      try {
        $this->connection
          ->delete($this->tableName)
          ->execute();
      } catch (\Exception $e) {
        $this
          ->ensureTableExists();
      }

      // Split the routes into chunks to avoid big INSERT queries.
      $route_chunks = array_chunk($this->routes
        ->all(), 50, TRUE);
      foreach ($route_chunks as $routes) {
        $insert = $this->connection
          ->insert($this->tableName)
          ->fields([
          'name',
          'fit',
          'path',
          'pattern_outline',
          'number_parts',
          'route',
        ]);
        $names = [];
        foreach ($routes as $name => $route) {

          /** @var \Symfony\Component\Routing\Route $route */
          $route
            ->setOption('compiler_class', RouteCompiler::class);

          /** @var \Drupal\Core\Routing\CompiledRoute $compiled */
          $compiled = $route
            ->compile();

          // The fit value is a binary number which has 1 at every fixed path
          // position and 0 where there is a wildcard. We keep track of all such
          // patterns that exist so that we can minimize the number of path
          // patterns we need to check in the RouteProvider.
          $masks[$compiled
            ->getFit()] = 1;
          $names[] = $name;
          $values = [
            'name' => $name,
            'fit' => $compiled
              ->getFit(),
            'path' => $route
              ->getPath(),
            'pattern_outline' => $compiled
              ->getPatternOutline(),
            'number_parts' => $compiled
              ->getNumParts(),
            'route' => serialize($route),
          ];
          $insert
            ->values($values);
        }

        // Insert all new routes.
        $insert
          ->execute();
      }
    } catch (\Exception $e) {
      $transaction
        ->rollBack();
      watchdog_exception('Routing', $e);
      throw $e;
    }

    // Sort the masks so they are in order of descending fit.
    $masks = array_keys($masks);
    rsort($masks);
    $this->state
      ->set('routing.menu_masks.' . $this->tableName, $masks);
    $this->routes = NULL;
  }

  /**
   * Gets the routes to match.
   *
   * @return \Symfony\Component\Routing\RouteCollection
   *   A RouteCollection instance representing all routes currently in the
   *   dumper.
   */
  public function getRoutes() {
    return $this->routes;
  }

  /**
   * Checks if the tree table exists and create it if not.
   *
   * @return bool
   *   TRUE if the table was created, FALSE otherwise.
   */
  protected function ensureTableExists() {
    try {
      if (!$this->connection
        ->schema()
        ->tableExists($this->tableName)) {
        $this->connection
          ->schema()
          ->createTable($this->tableName, $this
          ->schemaDefinition());
        return TRUE;
      }
    } catch (DatabaseException $e) {

      // If another process has already created the config table, attempting to
      // recreate it will throw an exception. In this case just catch the
      // exception and do nothing.
      return TRUE;
    }
    return FALSE;
  }

  /**
   * Defines the schema for the router table.
   *
   * @return array
   *   The schema API definition for the SQL storage table.
   *
   * @internal
   */
  protected function schemaDefinition() {
    $schema = [
      'description' => 'Maps paths to various callbacks (access, page and title)',
      'fields' => [
        'name' => [
          'description' => 'Primary Key: Machine name of this route',
          'type' => 'varchar_ascii',
          'length' => 255,
          'not null' => TRUE,
          'default' => '',
        ],
        'path' => [
          'description' => 'The path for this URI',
          'type' => 'varchar',
          'length' => 255,
          'not null' => TRUE,
          'default' => '',
        ],
        'pattern_outline' => [
          'description' => 'The pattern',
          'type' => 'varchar',
          'length' => 255,
          'not null' => TRUE,
          'default' => '',
        ],
        'fit' => [
          'description' => 'A numeric representation of how specific the path is.',
          'type' => 'int',
          'not null' => TRUE,
          'default' => 0,
        ],
        'route' => [
          'description' => 'A serialized Route object',
          'type' => 'blob',
          'size' => 'big',
        ],
        'number_parts' => [
          'description' => 'Number of parts in this router path.',
          'type' => 'int',
          'not null' => TRUE,
          'default' => 0,
          'size' => 'small',
        ],
      ],
      'indexes' => [
        'pattern_outline_parts' => [
          'pattern_outline',
          'number_parts',
        ],
      ],
      'primary key' => [
        'name',
      ],
    ];
    return $schema;
  }

}

Members

Namesort descending Modifiers Type Description Overrides
MatcherDumper::$connection protected property The database connection to which to dump route information.
MatcherDumper::$routes protected property The routes to be dumped.
MatcherDumper::$state protected property The state.
MatcherDumper::$tableName protected property The name of the SQL table to which to dump the routes.
MatcherDumper::addRoutes public function Adds additional routes to be dumped. Overrides MatcherDumperInterface::addRoutes
MatcherDumper::dump public function Dumps a set of routes to the router table in the database.
MatcherDumper::ensureTableExists protected function Checks if the tree table exists and create it if not.
MatcherDumper::getRoutes public function Gets the routes to match.
MatcherDumper::schemaDefinition protected function Defines the schema for the router table.
MatcherDumper::__construct public function Construct the MatcherDumper.