MenuTreeStorage.php
Same filename in other branches
Namespace
Drupal\Core\MenuFile
-
core/
lib/ Drupal/ Core/ Menu/ MenuTreeStorage.php
View source
<?php
namespace Drupal\Core\Menu;
use Drupal\Component\Plugin\Exception\PluginException;
use Drupal\Component\Utility\UrlHelper;
use Drupal\Core\Cache\Cache;
use Drupal\Core\Cache\CacheBackendInterface;
use Drupal\Core\Cache\CacheTagsInvalidatorInterface;
use Drupal\Core\Database\Connection;
use Drupal\Core\Database\Database;
use Drupal\Core\Database\DatabaseException;
use Drupal\Core\Database\Query\SelectInterface;
// cspell:ignore mlid
/**
* Provides a menu tree storage using the database.
*/
class MenuTreeStorage implements MenuTreeStorageInterface {
use MenuLinkFieldDefinitions;
/**
* The maximum depth of a menu links tree.
*/
const MAX_DEPTH = 9;
/**
* The database connection.
*
* @var \Drupal\Core\Database\Connection
*/
protected $connection;
/**
* Cache backend instance for the extracted tree data.
*
* @var \Drupal\Core\Cache\CacheBackendInterface
*/
protected $menuCacheBackend;
/**
* The cache tags invalidator.
*
* @var \Drupal\Core\Cache\CacheTagsInvalidatorInterface
*/
protected $cacheTagsInvalidator;
/**
* The database table name.
*
* @var string
*/
protected $table;
/**
* Additional database connection options to use in queries.
*
* @var array
*/
protected $options = [];
/**
* Stores definitions that have already been loaded for better performance.
*
* An array of plugin definition arrays, keyed by plugin ID.
*
* @var array
*/
protected $definitions = [];
/**
* List of serialized fields.
*
* @var array
*/
protected $serializedFields;
/**
* Constructs a new \Drupal\Core\Menu\MenuTreeStorage.
*
* @param \Drupal\Core\Database\Connection $connection
* A Database connection to use for reading and writing configuration data.
* @param \Drupal\Core\Cache\CacheBackendInterface $menu_cache_backend
* Cache backend instance for the extracted tree data.
* @param \Drupal\Core\Cache\CacheTagsInvalidatorInterface $cache_tags_invalidator
* The cache tags invalidator.
* @param string $table
* A database table name to store configuration data in.
* @param array $options
* (optional) Any additional database connection options to use in queries.
*/
public function __construct(Connection $connection, CacheBackendInterface $menu_cache_backend, CacheTagsInvalidatorInterface $cache_tags_invalidator, $table, array $options = []) {
$this->connection = $connection;
$this->menuCacheBackend = $menu_cache_backend;
$this->cacheTagsInvalidator = $cache_tags_invalidator;
$this->table = $table;
$this->options = $options;
}
/**
* {@inheritdoc}
*/
public function maxDepth() {
return static::MAX_DEPTH;
}
/**
* {@inheritdoc}
*/
public function resetDefinitions() {
$this->definitions = [];
}
/**
* {@inheritdoc}
*/
public function rebuild(array $definitions) {
$links = [];
$children = [];
$top_links = [];
// Fetch the list of existing menus, in case some are not longer populated
// after the rebuild.
$before_menus = $this->getMenuNames();
if ($definitions) {
foreach ($definitions as $id => $link) {
// Flag this link as discovered, i.e. saved via rebuild().
$link['discovered'] = 1;
// Note: The parent we set here might be just stored in the {menu_tree}
// table, so it will not end up in $top_links. Therefore the later loop
// on the orphan links, will handle those cases.
if (!empty($link['parent'])) {
$children[$link['parent']][$id] = $id;
}
else {
// A top level link - we need them to root our tree.
$top_links[$id] = $id;
$link['parent'] = '';
}
$links[$id] = $link;
}
}
foreach ($top_links as $id) {
$this->saveRecursive($id, $children, $links);
}
// Handle any children we didn't find starting from top-level links.
foreach ($children as $orphan_links) {
foreach ($orphan_links as $id) {
// Check for a parent that is not loaded above since only internal links
// are loaded above.
$parent = $this->loadFull($links[$id]['parent']);
// If there is a parent add it to the links to be used in
// ::saveRecursive().
if ($parent) {
$links[$links[$id]['parent']] = $parent;
}
else {
// Force it to the top level.
$links[$id]['parent'] = '';
}
$this->saveRecursive($id, $children, $links);
}
}
$result = $this->findNoLongerExistingLinks($definitions);
// Remove all such items.
if ($result) {
$this->purgeMultiple($result);
}
$this->resetDefinitions();
$affected_menus = $this->getMenuNames() + $before_menus;
// Invalidate any cache tagged with any menu name.
$cache_tags = Cache::buildTags('config:system.menu', $affected_menus, '.');
$this->cacheTagsInvalidator
->invalidateTags($cache_tags);
$this->resetDefinitions();
// Every item in the cache bin should have one of the menu cache tags but it
// is not guaranteed, so invalidate everything in the bin.
$this->menuCacheBackend
->invalidateAll();
}
/**
* Purges multiple menu links that no longer exist.
*
* @param array $ids
* An array of menu link IDs.
*/
protected function purgeMultiple(array $ids) {
$loaded = $this->loadFullMultiple($ids);
foreach ($loaded as $id => $link) {
if ($link['has_children']) {
$children = $this->loadByProperties([
'parent' => $id,
]);
foreach ($children as $child) {
$child['parent'] = $link['parent'];
$this->save($child);
}
}
}
$this->doDeleteMultiple($ids);
}
/**
* Executes a select query while making sure the database table exists.
*
* @param \Drupal\Core\Database\Query\SelectInterface $query
* The select object to be executed.
*
* @return \Drupal\Core\Database\StatementInterface|null
* A prepared statement, or NULL if the query is not valid.
*
* @throws \Exception
* Thrown if the table could not be created or the database connection
* failed.
*/
protected function safeExecuteSelect(SelectInterface $query) {
try {
return $query->execute();
} catch (\Exception $e) {
// If there was an exception, try to create the table.
if ($this->ensureTableExists()) {
return $query->execute();
}
// Some other failure that we can not recover from.
throw new PluginException($e->getMessage(), 0, $e);
}
}
/**
* {@inheritdoc}
*/
public function save(array $link) {
$affected_menus = $this->doSave($link);
$this->resetDefinitions();
$cache_tags = Cache::buildTags('config:system.menu', $affected_menus, '.');
$this->cacheTagsInvalidator
->invalidateTags($cache_tags);
return $affected_menus;
}
/**
* Saves a link without clearing caches.
*
* @param array $link
* A definition, according to $definitionFields, for a
* \Drupal\Core\Menu\MenuLinkInterface plugin.
*
* @return array
* The menu names affected by the save operation. This will be one menu
* name if the link is saved to the sane menu, or two if it is saved to a
* new menu.
*
* @throws \Exception
* Thrown if the storage back-end does not exist and could not be created.
* @throws \Drupal\Component\Plugin\Exception\PluginException
* Thrown if the definition is invalid, for example, if the specified parent
* would cause the links children to be moved to greater than the maximum
* depth.
*/
protected function doSave(array $link) {
$affected_menus = [];
// Get the existing definition if it exists. This does not use
// self::loadFull() to avoid the unserialization of fields with 'serialize'
// equal to TRUE as defined in self::schemaDefinition(). The makes $original
// easier to compare with the return value of self::preSave().
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->fields($this->table);
$query->condition('id', $link['id']);
$original = $this->safeExecuteSelect($query)
->fetchAssoc();
if ($original) {
$link['mlid'] = $original['mlid'];
$link['has_children'] = $original['has_children'];
$affected_menus[$original['menu_name']] = $original['menu_name'];
$fields = $this->preSave($link, $original);
// If $link matches the $original data then exit early as there are no
// changes to make. Use array_diff_assoc() to check if they match because:
// - Some of the data types of the values are not the same. The values
// in $original are all strings because they have come from database but
// $fields contains typed values.
// - MenuTreeStorage::preSave() removes the 'mlid' from $fields.
// - The order of the keys in $original and $fields is different.
if (array_diff_assoc($fields, $original) == [] && array_diff_assoc($original, $fields) == [
'mlid' => $link['mlid'],
]) {
return $affected_menus;
}
}
try {
$transaction = $this->connection
->startTransaction();
if (!$original) {
// Generate a new mlid.
// @todo Remove the 'return' option in Drupal 11.
// @see https://www.drupal.org/project/drupal/issues/3256524
$options = [
'return' => Database::RETURN_INSERT_ID,
] + $this->options;
$link['mlid'] = $this->connection
->insert($this->table, $options)
->fields([
'id' => $link['id'],
'menu_name' => $link['menu_name'],
])
->execute();
$fields = $this->preSave($link, []);
}
// We may be moving the link to a new menu.
$affected_menus[$fields['menu_name']] = $fields['menu_name'];
$query = $this->connection
->update($this->table, $this->options);
$query->condition('mlid', $link['mlid']);
$query->fields($fields)
->execute();
if ($original) {
$this->updateParentalStatus($original);
}
$this->updateParentalStatus($link);
} catch (\Exception $e) {
if (isset($transaction)) {
$transaction->rollBack();
}
throw $e;
}
return $affected_menus;
}
/**
* Fills in all the fields the database save needs, using the link definition.
*
* @param array $link
* The link definition to be updated.
* @param array $original
* The link definition before the changes. May be empty if not found.
*
* @return array
* The values which will be stored.
*
* @throws \Drupal\Component\Plugin\Exception\PluginException
* Thrown when the specific depth exceeds the maximum.
*/
protected function preSave(array &$link, array $original) {
static $schema_fields, $schema_defaults;
if (empty($schema_fields)) {
$schema = static::schemaDefinition();
$schema_fields = $schema['fields'];
foreach ($schema_fields as $name => $spec) {
if (isset($spec['default'])) {
$schema_defaults[$name] = $spec['default'];
}
}
}
// Try to find a parent link. If found, assign it and derive its menu.
$parent = $this->findParent($link, $original);
if ($parent) {
$link['parent'] = $parent['id'];
$link['menu_name'] = $parent['menu_name'];
}
else {
$link['parent'] = '';
}
// If no corresponding parent link was found, move the link to the
// top-level.
foreach ($schema_defaults as $name => $default) {
if (!isset($link[$name])) {
$link[$name] = $default;
}
}
$fields = array_intersect_key($link, $schema_fields);
// Sort the route parameters so that the query string will be the same.
asort($fields['route_parameters']);
// Since this will be urlencoded, it's safe to store and match against a
// text field.
$fields['route_param_key'] = $fields['route_parameters'] ? UrlHelper::buildQuery($fields['route_parameters']) : '';
foreach ($this->serializedFields() as $name) {
if (isset($fields[$name])) {
$fields[$name] = serialize($fields[$name]);
}
}
$this->setParents($fields, $parent, $original);
// Need to check both parent and menu_name, since parent can be empty in any
// menu.
if ($original && ($link['parent'] != $original['parent'] || $link['menu_name'] != $original['menu_name'])) {
$this->moveChildren($fields, $original);
}
// We needed the mlid above, but not in the update query.
unset($fields['mlid']);
// Cast Booleans to int, if needed.
$fields['enabled'] = (int) $fields['enabled'];
$fields['expanded'] = (int) $fields['expanded'];
return $fields;
}
/**
* {@inheritdoc}
*/
public function delete($id) {
// Children get re-attached to the menu link's parent.
$item = $this->loadFull($id);
// It's possible the link is already deleted.
if ($item) {
$parent = $item['parent'];
$children = $this->loadByProperties([
'parent' => $id,
]);
foreach ($children as $child) {
$child['parent'] = $parent;
$this->save($child);
}
$this->doDeleteMultiple([
$id,
]);
$this->updateParentalStatus($item);
// Many children may have moved.
$this->resetDefinitions();
$this->cacheTagsInvalidator
->invalidateTags([
'config:system.menu.' . $item['menu_name'],
]);
}
}
/**
* {@inheritdoc}
*/
public function getSubtreeHeight($id) {
$original = $this->loadFull($id);
return $original ? $this->doFindChildrenRelativeDepth($original) + 1 : 0;
}
/**
* Finds the relative depth of this link's deepest child.
*
* @param array $original
* The parent definition used to find the depth.
*
* @return int
* Returns the relative depth.
*/
protected function doFindChildrenRelativeDepth(array $original) {
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->addField($this->table, 'depth');
$query->condition('menu_name', $original['menu_name']);
$query->orderBy('depth', 'DESC');
$query->range(0, 1);
for ($i = 1; $i <= static::MAX_DEPTH && $original["p{$i}"]; $i++) {
$query->condition("p{$i}", $original["p{$i}"]);
}
$max_depth = $this->safeExecuteSelect($query)
->fetchField();
return $max_depth > $original['depth'] ? $max_depth - $original['depth'] : 0;
}
/**
* Sets the materialized path field values based on the parent.
*
* @param array $fields
* The menu link.
* @param array|false $parent
* The parent menu link.
* @param array $original
* The original menu link.
*/
protected function setParents(array &$fields, $parent, array $original) {
// Directly fill parents for top-level links.
if (empty($fields['parent'])) {
$fields['p1'] = $fields['mlid'];
for ($i = 2; $i <= $this->maxDepth(); $i++) {
$fields["p{$i}"] = 0;
}
$fields['depth'] = 1;
}
else {
// @todo We want to also check $original['has_children'] here, but that
// will be 0 even if there are children if those are not enabled.
// has_children is really just the rendering hint. So, we either need
// to define another column (has_any_children), or do the extra query.
// https://www.drupal.org/node/2302149
if ($original) {
$limit = $this->maxDepth() - $this->doFindChildrenRelativeDepth($original) - 1;
}
else {
$limit = $this->maxDepth() - 1;
}
if ($parent['depth'] > $limit) {
throw new PluginException("The link with ID {$fields['id']} or its children exceeded the maximum depth of {$this->maxDepth()}");
}
$fields['depth'] = $parent['depth'] + 1;
$i = 1;
while ($i < $fields['depth']) {
$p = 'p' . $i++;
$fields[$p] = $parent[$p];
}
$p = 'p' . $i++;
// The parent (p1 - p9) corresponding to the depth always equals the mlid.
$fields[$p] = $fields['mlid'];
while ($i <= static::MAX_DEPTH) {
$p = 'p' . $i++;
$fields[$p] = 0;
}
}
}
/**
* Re-parents a link's children when the link itself is moved.
*
* @param array $fields
* The changed menu link.
* @param array $original
* The original menu link.
*/
protected function moveChildren($fields, $original) {
$query = $this->connection
->update($this->table, $this->options);
$query->fields([
'menu_name' => $fields['menu_name'],
]);
$expressions = [];
for ($i = 1; $i <= $fields['depth']; $i++) {
$expressions[] = [
"p{$i}",
":p_{$i}",
[
":p_{$i}" => $fields["p{$i}"],
],
];
}
$j = $original['depth'] + 1;
while ($i <= $this->maxDepth() && $j <= $this->maxDepth()) {
$expressions[] = [
'p' . $i++,
'p' . $j++,
[],
];
}
while ($i <= $this->maxDepth()) {
$expressions[] = [
'p' . $i++,
0,
[],
];
}
$shift = $fields['depth'] - $original['depth'];
if ($shift > 0) {
// The order of expressions must be reversed so the new values don't
// overwrite the old ones before they can be used because "Single-table
// UPDATE assignments are generally evaluated from left to right".
// @see http://dev.mysql.com/doc/refman/5.0/en/update.html
$expressions = array_reverse($expressions);
}
foreach ($expressions as $expression) {
$query->expression($expression[0], $expression[1], $expression[2]);
}
$query->expression('depth', '[depth] + :depth', [
':depth' => $shift,
]);
$query->condition('menu_name', $original['menu_name']);
for ($i = 1; $i <= $this->maxDepth() && $original["p{$i}"]; $i++) {
$query->condition("p{$i}", $original["p{$i}"]);
}
$query->execute();
}
/**
* Loads the parent definition if it exists.
*
* @param array $link
* The link definition to find the parent of.
* @param array|false $original
* The original link that might be used to find the parent if the parent
* is not set on the $link, or FALSE if the original could not be loaded.
*
* @return array|false
* Returns a definition array, or FALSE if no parent was found.
*/
protected function findParent($link, $original) {
$parent = FALSE;
// This item is explicitly top-level, skip the rest of the parenting.
if (isset($link['parent']) && empty($link['parent'])) {
return $parent;
}
// If we have a parent link ID, try to use that.
$candidates = [];
if (isset($link['parent'])) {
$candidates[] = $link['parent'];
}
elseif (!empty($original['parent']) && $link['menu_name'] == $original['menu_name']) {
// Otherwise, fall back to the original parent.
$candidates[] = $original['parent'];
}
foreach ($candidates as $id) {
$parent = $this->loadFull($id);
if ($parent) {
break;
}
}
return $parent;
}
/**
* Sets has_children for the link's parent if it has visible children.
*
* @param array $link
* The link to get a parent ID from.
*/
protected function updateParentalStatus(array $link) {
// If parent is empty, there is nothing to update.
if (!empty($link['parent'])) {
// Check if at least one visible child exists in the table.
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->addExpression('1');
$query->range(0, 1);
$query->condition('menu_name', $link['menu_name'])
->condition('parent', $link['parent'])
->condition('enabled', 1);
$parent_has_children = (bool) $query->execute()
->fetchField() ? 1 : 0;
$this->connection
->update($this->table, $this->options)
->fields([
'has_children' => $parent_has_children,
])
->condition('id', $link['parent'])
->execute();
}
}
/**
* Prepares a link by unserializing values and saving the definition.
*
* @param array $link
* The data loaded in the query.
* @param bool $intersect
* If TRUE, filter out values that are not part of the actual definition.
*
* @return array
* The prepared link data.
*/
protected function prepareLink(array $link, $intersect = FALSE) {
foreach ($this->serializedFields() as $name) {
if (isset($link[$name])) {
$link[$name] = unserialize($link[$name]);
}
}
if ($intersect) {
$link = array_intersect_key($link, array_flip($this->definitionFields()));
}
$this->definitions[$link['id']] = $link;
return $link;
}
/**
* {@inheritdoc}
*/
public function loadByProperties(array $properties) {
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->fields($this->table, $this->definitionFields());
foreach ($properties as $name => $value) {
if (!in_array($name, $this->definitionFields(), TRUE)) {
$fields = implode(', ', $this->definitionFields());
throw new \InvalidArgumentException("An invalid property name, {$name} was specified. Allowed property names are: {$fields}.");
}
$query->condition($name, $value);
}
$loaded = $this->safeExecuteSelect($query)
->fetchAllAssoc('id', \PDO::FETCH_ASSOC);
foreach ($loaded as $id => $link) {
$loaded[$id] = $this->prepareLink($link);
}
return $loaded;
}
/**
* {@inheritdoc}
*/
public function loadByRoute($route_name, array $route_parameters = [], $menu_name = NULL) {
// Sort the route parameters so that the query string will be the same.
asort($route_parameters);
// Since this will be urlencoded, it's safe to store and match against a
// text field.
// @todo Standardize an efficient way to load by route name and parameters
// in place of system path. https://www.drupal.org/node/2302139
$param_key = $route_parameters ? UrlHelper::buildQuery($route_parameters) : '';
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->fields($this->table, $this->definitionFields());
$query->condition('route_name', $route_name);
$query->condition('route_param_key', $param_key);
if ($menu_name) {
$query->condition('menu_name', $menu_name);
}
// Make the ordering deterministic.
$query->orderBy('depth');
$query->orderBy('weight');
$query->orderBy('id');
$loaded = $this->safeExecuteSelect($query)
->fetchAllAssoc('id', \PDO::FETCH_ASSOC);
foreach ($loaded as $id => $link) {
$loaded[$id] = $this->prepareLink($link);
}
return $loaded;
}
/**
* {@inheritdoc}
*/
public function loadMultiple(array $ids) {
$missing_ids = array_diff($ids, array_keys($this->definitions));
if ($missing_ids) {
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->fields($this->table, $this->definitionFields());
$query->condition('id', $missing_ids, 'IN');
$loaded = $this->safeExecuteSelect($query)
->fetchAllAssoc('id', \PDO::FETCH_ASSOC);
foreach ($loaded as $id => $link) {
$this->definitions[$id] = $this->prepareLink($link);
}
}
return array_intersect_key($this->definitions, array_flip($ids));
}
/**
* {@inheritdoc}
*/
public function load($id) {
if (isset($this->definitions[$id])) {
return $this->definitions[$id];
}
$loaded = $this->loadMultiple([
$id,
]);
return $loaded[$id] ?? FALSE;
}
/**
* Loads all table fields, not just those that are in the plugin definition.
*
* @param string $id
* The menu link ID.
*
* @return array
* The loaded menu link definition or an empty array if not be found.
*/
protected function loadFull($id) {
$loaded = $this->loadFullMultiple([
$id,
]);
return $loaded[$id] ?? [];
}
/**
* Loads all table fields for multiple menu link definitions by ID.
*
* @param array $ids
* The IDs to load.
*
* @return array
* The loaded menu link definitions.
*/
protected function loadFullMultiple(array $ids) {
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->fields($this->table);
$query->condition('id', $ids, 'IN');
$loaded = $this->safeExecuteSelect($query)
->fetchAllAssoc('id', \PDO::FETCH_ASSOC);
foreach ($loaded as &$link) {
foreach ($this->serializedFields() as $name) {
if (isset($link[$name])) {
$link[$name] = unserialize($link[$name]);
}
}
}
return $loaded;
}
/**
* {@inheritdoc}
*/
public function getRootPathIds($id) {
$subquery = $this->connection
->select($this->table, NULL, $this->options);
// @todo Consider making this dynamic based on static::MAX_DEPTH or from the
// schema if that is generated using static::MAX_DEPTH.
// https://www.drupal.org/node/2302043
$subquery->fields($this->table, [
'p1',
'p2',
'p3',
'p4',
'p5',
'p6',
'p7',
'p8',
'p9',
]);
$subquery->condition('id', $id);
$result = current($subquery->execute()
->fetchAll(\PDO::FETCH_ASSOC));
$ids = array_filter($result);
if ($ids) {
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->fields($this->table, [
'id',
]);
$query->orderBy('depth', 'DESC');
$query->condition('mlid', $ids, 'IN');
// @todo Cache this result in memory if we find it is being used more
// than once per page load. https://www.drupal.org/node/2302185
return $this->safeExecuteSelect($query)
->fetchAllKeyed(0, 0);
}
return [];
}
/**
* {@inheritdoc}
*/
public function getExpanded($menu_name, array $parents) {
// @todo Go back to tracking in state or some other way which menus have
// expanded links? https://www.drupal.org/node/2302187
do {
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->fields($this->table, [
'id',
]);
$query->condition('menu_name', $menu_name);
$query->condition('expanded', 1);
$query->condition('has_children', 1);
$query->condition('enabled', 1);
$query->condition('parent', $parents, 'IN');
$query->condition('id', $parents, 'NOT IN');
$result = $this->safeExecuteSelect($query)
->fetchAllKeyed(0, 0);
$parents += $result;
} while (!empty($result));
return $parents;
}
/**
* Saves menu links recursively.
*
* @param string $id
* The definition ID.
* @param array $children
* An array of IDs of child links collected by parent ID.
* @param array $links
* An array of all definitions keyed by ID.
*/
protected function saveRecursive($id, &$children, &$links) {
if (!empty($links[$id]['parent']) && empty($links[$links[$id]['parent']])) {
// Invalid parent ID, so remove it.
$links[$id]['parent'] = '';
}
$this->doSave($links[$id]);
if (!empty($children[$id])) {
foreach ($children[$id] as $next_id) {
$this->saveRecursive($next_id, $children, $links);
}
}
// Remove processed link names so we can find stragglers.
unset($children[$id]);
}
/**
* {@inheritdoc}
*/
public function loadTreeData($menu_name, MenuTreeParameters $parameters) {
$tree_cid = "tree-data:{$menu_name}:" . serialize($parameters);
$cache = $this->menuCacheBackend
->get($tree_cid);
if ($cache && isset($cache->data)) {
$data = $cache->data;
// Cache the definitions in memory so they don't need to be loaded again.
$this->definitions += $data['definitions'];
unset($data['definitions']);
}
else {
$links = $this->loadLinks($menu_name, $parameters);
$data['tree'] = $this->doBuildTreeData($links, $parameters->activeTrail, $parameters->minDepth);
$data['definitions'] = [];
$data['route_names'] = $this->collectRoutesAndDefinitions($data['tree'], $data['definitions']);
$this->menuCacheBackend
->set($tree_cid, $data, Cache::PERMANENT, [
'config:system.menu.' . $menu_name,
]);
// The definitions were already added to $this->definitions in
// $this->doBuildTreeData()
unset($data['definitions']);
}
return $data;
}
/**
* Loads links in the given menu, according to the given tree parameters.
*
* @param string $menu_name
* A menu name.
* @param \Drupal\Core\Menu\MenuTreeParameters $parameters
* The parameters to determine which menu links to be loaded into a tree.
* This method will set the absolute minimum depth, which is used in
* MenuTreeStorage::doBuildTreeData().
*
* @return array
* A flat array of menu links that are part of the menu. Each array element
* is an associative array of information about the menu link, containing
* the fields from the {menu_tree} table. This array must be ordered
* depth-first.
*/
protected function loadLinks($menu_name, MenuTreeParameters $parameters) {
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->fields($this->table);
// Allow a custom root to be specified for loading a menu link tree. If
// omitted, the default root (i.e. the actual root, '') is used.
if ($parameters->root !== '') {
$root = $this->loadFull($parameters->root);
// If the custom root does not exist, we cannot load the links below it.
if (!$root) {
return [];
}
// When specifying a custom root, we only want to find links whose
// parent IDs match that of the root; that's how we ignore the rest of the
// tree. In other words: we exclude everything unreachable from the
// custom root.
for ($i = 1; $i <= $root['depth']; $i++) {
$query->condition("p{$i}", $root["p{$i}"]);
}
// When specifying a custom root, the menu is determined by that root.
$menu_name = $root['menu_name'];
// If the custom root exists, then we must rewrite some of our
// parameters; parameters are relative to the root (default or custom),
// but the queries require absolute numbers, so adjust correspondingly.
if (isset($parameters->minDepth)) {
$parameters->minDepth += $root['depth'];
}
else {
$parameters->minDepth = $root['depth'];
}
if (isset($parameters->maxDepth)) {
$parameters->maxDepth += $root['depth'];
}
}
// If no minimum depth is specified, then set the actual minimum depth,
// depending on the root.
if (!isset($parameters->minDepth)) {
if ($parameters->root !== '' && $root) {
$parameters->minDepth = $root['depth'];
}
else {
$parameters->minDepth = 1;
}
}
for ($i = 1; $i <= $this->maxDepth(); $i++) {
$query->orderBy('p' . $i, 'ASC');
}
$query->condition('menu_name', $menu_name);
if (!empty($parameters->expandedParents)) {
$query->condition('parent', $parameters->expandedParents, 'IN');
}
if (isset($parameters->minDepth) && $parameters->minDepth > 1) {
$query->condition('depth', $parameters->minDepth, '>=');
}
if (isset($parameters->maxDepth)) {
$query->condition('depth', $parameters->maxDepth, '<=');
}
// Add custom query conditions, if any were passed.
if (!empty($parameters->conditions)) {
// Only allow conditions that are testing definition fields.
$parameters->conditions = array_intersect_key($parameters->conditions, array_flip($this->definitionFields()));
$serialized_fields = $this->serializedFields();
foreach ($parameters->conditions as $column => $value) {
if (is_array($value)) {
$operator = $value[1];
$value = $value[0];
}
else {
$operator = '=';
}
if (in_array($column, $serialized_fields)) {
$value = serialize($value);
}
$query->condition($column, $value, $operator);
}
}
$links = $this->safeExecuteSelect($query)
->fetchAllAssoc('id', \PDO::FETCH_ASSOC);
return $links;
}
/**
* Traverses the menu tree and collects all the route names and definitions.
*
* @param array $tree
* The menu tree you wish to operate on.
* @param array $definitions
* An array to accumulate definitions by reference.
*
* @return array
* Array of route names, with all values being unique.
*/
protected function collectRoutesAndDefinitions(array $tree, array &$definitions) {
return array_values($this->doCollectRoutesAndDefinitions($tree, $definitions));
}
/**
* Collects all the route names and definitions.
*
* @param array $tree
* A menu link tree from MenuTreeStorage::doBuildTreeData()
* @param array $definitions
* The collected definitions which are populated by reference.
*
* @return array
* The collected route names.
*/
protected function doCollectRoutesAndDefinitions(array $tree, array &$definitions) {
$route_names = [];
foreach (array_keys($tree) as $id) {
$definitions[$id] = $this->definitions[$id];
if (!empty($definitions[$id]['route_name'])) {
$route_names[$definitions[$id]['route_name']] = $definitions[$id]['route_name'];
}
if ($tree[$id]['subtree']) {
$route_names += $this->doCollectRoutesAndDefinitions($tree[$id]['subtree'], $definitions);
}
}
return $route_names;
}
/**
* {@inheritdoc}
*/
public function loadSubtreeData($id, $max_relative_depth = NULL) {
$tree = [];
$root = $this->loadFull($id);
if (!$root) {
return $tree;
}
$parameters = new MenuTreeParameters();
$parameters->setRoot($id)
->onlyEnabledLinks();
return $this->loadTreeData($root['menu_name'], $parameters);
}
/**
* {@inheritdoc}
*/
public function menuNameInUse($menu_name) {
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->addField($this->table, 'mlid');
$query->condition('menu_name', $menu_name);
$query->range(0, 1);
return (bool) $this->safeExecuteSelect($query);
}
/**
* {@inheritdoc}
*/
public function getMenuNames() {
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->addField($this->table, 'menu_name');
$query->distinct();
return $this->safeExecuteSelect($query)
->fetchAllKeyed(0, 0);
}
/**
* {@inheritdoc}
*/
public function countMenuLinks($menu_name = NULL) {
$query = $this->connection
->select($this->table, NULL, $this->options);
if ($menu_name) {
$query->condition('menu_name', $menu_name);
}
return $this->safeExecuteSelect($query->countQuery())
->fetchField();
}
/**
* {@inheritdoc}
*/
public function getAllChildIds($id) {
$root = $this->loadFull($id);
if (!$root) {
return [];
}
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->fields($this->table, [
'id',
]);
$query->condition('menu_name', $root['menu_name']);
for ($i = 1; $i <= $root['depth']; $i++) {
$query->condition("p{$i}", $root["p{$i}"]);
}
// The next p column should not be empty. This excludes the root link.
$query->condition("p{$i}", 0, '>');
return $this->safeExecuteSelect($query)
->fetchAllKeyed(0, 0);
}
/**
* {@inheritdoc}
*/
public function loadAllChildren($id, $max_relative_depth = NULL) {
$parameters = new MenuTreeParameters();
$parameters->setRoot($id)
->excludeRoot()
->setMaxDepth($max_relative_depth)
->onlyEnabledLinks();
$links = $this->loadLinks(NULL, $parameters);
foreach ($links as $id => $link) {
$links[$id] = $this->prepareLink($link);
}
return $links;
}
/**
* Prepares the data for calling $this->treeDataRecursive().
*/
protected function doBuildTreeData(array $links, array $parents = [], $depth = 1) {
// Reverse the array so we can use the more efficient array_pop() function.
$links = array_reverse($links);
return $this->treeDataRecursive($links, $parents, $depth);
}
/**
* Builds the data representing a menu tree.
*
* The function is a bit complex because the rendering of a link depends on
* the next menu link.
*
* @param array $links
* A flat array of menu links that are part of the menu. Each array element
* is an associative array of information about the menu link, containing
* the fields from the $this->table. This array must be ordered
* depth-first. MenuTreeStorage::loadTreeData() includes a sample query.
* @param array $parents
* An array of the menu link ID values that are in the path from the current
* page to the root of the menu tree.
* @param int $depth
* The minimum depth to include in the returned menu tree.
*
* @return array
* The fully built tree.
*
* @see \Drupal\Core\Menu\MenuTreeStorage::loadTreeData()
*/
protected function treeDataRecursive(array &$links, array $parents, $depth) {
$tree = [];
while ($tree_link_definition = array_pop($links)) {
$tree[$tree_link_definition['id']] = [
'definition' => $this->prepareLink($tree_link_definition, TRUE),
'has_children' => $tree_link_definition['has_children'],
// We need to determine if we're on the path to root so we can later
// build the correct active trail.
'in_active_trail' => in_array($tree_link_definition['id'], $parents),
'subtree' => [],
'depth' => $tree_link_definition['depth'],
];
// Look ahead to the next link, but leave it on the array so it's
// available to other recursive function calls if we return or build a
// sub-tree.
$next = end($links);
// Check whether the next link is the first in a new sub-tree.
if ($next && $next['depth'] > $depth) {
// Recursively call doBuildTreeData to build the sub-tree.
$tree[$tree_link_definition['id']]['subtree'] = $this->treeDataRecursive($links, $parents, $next['depth']);
// Fetch next link after filling the sub-tree.
$next = end($links);
}
// Determine if we should exit the loop and return.
if (!$next || $next['depth'] < $depth) {
break;
}
}
return $tree;
}
/**
* 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 {
$this->connection
->schema()
->createTable($this->table, static::schemaDefinition());
} 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.
} catch (\Exception $e) {
return FALSE;
}
return TRUE;
}
/**
* Determines serialized fields in the storage.
*
* @return array
* A list of fields that are serialized in the database.
*/
protected function serializedFields() {
if (empty($this->serializedFields)) {
$schema = static::schemaDefinition();
foreach ($schema['fields'] as $name => $field) {
if (!empty($field['serialize'])) {
$this->serializedFields[] = $name;
}
}
}
return $this->serializedFields;
}
/**
* Determines fields that are part of the plugin definition.
*
* @return string[]
* The list of the subset of fields that are part of the plugin definition.
*/
protected function definitionFields() {
return array_keys($this->defaults);
}
/**
* Defines the schema for the tree table.
*
* @return array
* The schema API definition for the SQL storage table.
*
* @internal
*/
protected static function schemaDefinition() {
$schema = [
'description' => 'Contains the menu tree hierarchy.',
'fields' => [
'menu_name' => [
'description' => "The menu name. All links with the same menu name (such as 'tools') are part of the same menu.",
'type' => 'varchar_ascii',
'length' => 32,
'not null' => TRUE,
'default' => '',
],
'mlid' => [
'description' => 'The menu link ID (mlid) is the integer primary key.',
'type' => 'serial',
'unsigned' => TRUE,
'not null' => TRUE,
],
'id' => [
'description' => 'Unique machine name: the plugin ID.',
'type' => 'varchar_ascii',
'length' => 255,
'not null' => TRUE,
],
'parent' => [
'description' => 'The plugin ID for the parent of this link.',
'type' => 'varchar_ascii',
'length' => 255,
'not null' => TRUE,
'default' => '',
],
'route_name' => [
'description' => 'The machine name of a defined Symfony Route this menu link represents.',
'type' => 'varchar_ascii',
'length' => 255,
],
'route_param_key' => [
'description' => 'An encoded string of route parameters for loading by route.',
'type' => 'varchar',
'length' => 255,
],
'route_parameters' => [
'description' => 'Serialized array of route parameters of this menu link.',
'type' => 'blob',
'size' => 'big',
'not null' => FALSE,
'serialize' => TRUE,
],
'url' => [
'description' => 'The external path this link points to (when not using a route).',
'type' => 'varchar',
'length' => 255,
'not null' => TRUE,
'default' => '',
],
'title' => [
'description' => 'The serialized title for the link. May be a TranslatableMarkup.',
'type' => 'blob',
'size' => 'big',
'not null' => FALSE,
'serialize' => TRUE,
],
'description' => [
'description' => 'The serialized description of this link - used for admin pages and title attribute. May be a TranslatableMarkup.',
'type' => 'blob',
'size' => 'big',
'not null' => FALSE,
'serialize' => TRUE,
],
'class' => [
'description' => 'The class for this link plugin.',
'type' => 'text',
'not null' => FALSE,
],
'options' => [
'description' => 'A serialized array of URL options, such as a query string or HTML attributes.',
'type' => 'blob',
'size' => 'big',
'not null' => FALSE,
'serialize' => TRUE,
],
'provider' => [
'description' => 'The name of the module that generated this link.',
'type' => 'varchar_ascii',
'length' => DRUPAL_EXTENSION_NAME_MAX_LENGTH,
'not null' => TRUE,
'default' => 'system',
],
'enabled' => [
'description' => 'A flag for whether the link should be rendered in menus. (0 = a disabled menu link that may be shown on admin screens, 1 = a normal, visible link)',
'type' => 'int',
'not null' => TRUE,
'default' => 1,
'size' => 'small',
],
'discovered' => [
'description' => 'A flag for whether the link was discovered, so can be purged on rebuild',
'type' => 'int',
'not null' => TRUE,
'default' => 0,
'size' => 'small',
],
'expanded' => [
'description' => 'Flag for whether this link should be rendered as expanded in menus - expanded links always have their child links displayed, instead of only when the link is in the active trail (1 = expanded, 0 = not expanded)',
'type' => 'int',
'not null' => TRUE,
'default' => 0,
'size' => 'small',
],
'weight' => [
'description' => 'Link weight among links in the same menu at the same depth.',
'type' => 'int',
'not null' => TRUE,
'default' => 0,
],
'metadata' => [
'description' => 'A serialized array of data that may be used by the plugin instance.',
'type' => 'blob',
'size' => 'big',
'not null' => FALSE,
'serialize' => TRUE,
],
'has_children' => [
'description' => 'Flag indicating whether any enabled links have this link as a parent (1 = enabled children exist, 0 = no enabled children).',
'type' => 'int',
'not null' => TRUE,
'default' => 0,
'size' => 'small',
],
'depth' => [
'description' => 'The depth relative to the top level. A link with empty parent will have depth == 1.',
'type' => 'int',
'not null' => TRUE,
'default' => 0,
'size' => 'small',
],
'p1' => [
'description' => 'The first mlid in the materialized path. If N = depth, then pN must equal the mlid. If depth > 1 then p(N-1) must equal the parent link mlid. All pX where X > depth must equal zero. The columns p1 .. p9 are also called the parents.',
'type' => 'int',
'unsigned' => TRUE,
'not null' => TRUE,
'default' => 0,
],
'p2' => [
'description' => 'The second mlid in the materialized path. See p1.',
'type' => 'int',
'unsigned' => TRUE,
'not null' => TRUE,
'default' => 0,
],
'p3' => [
'description' => 'The third mlid in the materialized path. See p1.',
'type' => 'int',
'unsigned' => TRUE,
'not null' => TRUE,
'default' => 0,
],
'p4' => [
'description' => 'The fourth mlid in the materialized path. See p1.',
'type' => 'int',
'unsigned' => TRUE,
'not null' => TRUE,
'default' => 0,
],
'p5' => [
'description' => 'The fifth mlid in the materialized path. See p1.',
'type' => 'int',
'unsigned' => TRUE,
'not null' => TRUE,
'default' => 0,
],
'p6' => [
'description' => 'The sixth mlid in the materialized path. See p1.',
'type' => 'int',
'unsigned' => TRUE,
'not null' => TRUE,
'default' => 0,
],
'p7' => [
'description' => 'The seventh mlid in the materialized path. See p1.',
'type' => 'int',
'unsigned' => TRUE,
'not null' => TRUE,
'default' => 0,
],
'p8' => [
'description' => 'The eighth mlid in the materialized path. See p1.',
'type' => 'int',
'unsigned' => TRUE,
'not null' => TRUE,
'default' => 0,
],
'p9' => [
'description' => 'The ninth mlid in the materialized path. See p1.',
'type' => 'int',
'unsigned' => TRUE,
'not null' => TRUE,
'default' => 0,
],
'form_class' => [
'description' => 'meh',
'type' => 'varchar',
'length' => 255,
],
],
'indexes' => [
'menu_parents' => [
'menu_name',
'p1',
'p2',
'p3',
'p4',
'p5',
'p6',
'p7',
'p8',
'p9',
],
// @todo Test this index for effectiveness.
// https://www.drupal.org/node/2302197
'menu_parent_expand_child' => [
'menu_name',
'expanded',
'has_children',
[
'parent',
16,
],
],
'route_values' => [
[
'route_name',
32,
],
[
'route_param_key',
16,
],
],
],
'primary key' => [
'mlid',
],
'unique keys' => [
'id' => [
'id',
],
],
];
return $schema;
}
/**
* Find any previously discovered menu links that no longer exist.
*
* @param array $definitions
* The new menu link definitions.
*
* @return array
* A list of menu link IDs that no longer exist.
*/
protected function findNoLongerExistingLinks(array $definitions) {
if ($definitions) {
$query = $this->connection
->select($this->table, NULL, $this->options);
$query->addField($this->table, 'id');
$query->condition('discovered', 1);
$query->condition('id', array_keys($definitions), 'NOT IN');
// Starting from links with the greatest depth will minimize the amount
// of re-parenting done by the menu storage.
$query->orderBy('depth', 'DESC');
$result = $query->execute()
->fetchCol();
}
else {
$result = [];
}
return $result;
}
/**
* Purge menu links from the database.
*
* @param array $ids
* A list of menu link IDs to be purged.
*/
protected function doDeleteMultiple(array $ids) {
$this->connection
->delete($this->table, $this->options)
->condition('id', $ids, 'IN')
->execute();
}
}
Classes
Title | Deprecated | Summary |
---|---|---|
MenuTreeStorage | Provides a menu tree storage using the database. |
Buggy or inaccurate documentation? Please file an issue. Need support? Need help programming? Connect with the Drupal community.