From 559479f8db11cb76464b382a35af360d5ff95374 Mon Sep 17 00:00:00 2001
From: Nathaniel Catchpole <catch@35733.no-reply.drupal.org>
Date: Mon, 27 Jan 2014 11:55:37 +0000
Subject: [PATCH] Issue #2164367 by alexpott, tim.plunkett, dawehner: Rebuild
 router as few times as possible per request.

---
 core/core.services.yml                        | 11 ++-
 core/includes/common.inc                      |  1 -
 core/includes/menu.inc                        | 15 ++--
 .../RouterRebuildSubscriber.php               | 71 +++++++++++++++++++
 .../Drupal/Core/Extension/ThemeHandler.php    |  4 +-
 core/lib/Drupal/Core/Routing/RouteBuilder.php | 31 ++++++--
 .../Core/Routing/RouteBuilderInterface.php    | 35 +++++++++
 .../Core/Routing/RouteBuilderStatic.php       | 29 +++++++-
 .../lib/Drupal/Core/Routing/RouteProvider.php | 37 +++++++++-
 .../Core/Routing/RouteProviderInterface.php   |  5 ++
 .../block/Tests/BlockHiddenRegionTest.php     |  5 +-
 .../lib/Drupal/color/Tests/ColorTest.php      |  2 -
 .../Tests/ConfigTranslationUiTest.php         |  5 +-
 core/modules/field_ui/field_ui.module         |  8 +--
 .../Tests/Views/ExtensionViewsFieldTest.php   |  2 +-
 core/modules/menu/menu.install                |  1 -
 .../lib/Drupal/menu_link/Entity/MenuLink.php  |  2 +-
 .../menu_link/MenuLinkStorageController.php   |  2 +-
 .../lib/Drupal/search/Entity/SearchPage.php   | 14 ++--
 .../Tests/SearchKeywordsConditionsTest.php    |  2 -
 .../search/Tests/SearchPageOverrideTest.php   |  4 --
 .../system/lib/Drupal/system/Entity/Menu.php  |  1 +
 .../system/Tests/Menu/MenuRouterTest.php      |  1 -
 .../Drupal/system/Tests/Menu/RebuildTest.php  | 13 ++--
 .../Tests/Routing/MockRouteProvider.php       |  7 ++
 .../Tests/Routing/RouteProviderTest.php       | 33 +++++----
 core/modules/system/system.admin.inc          |  4 +-
 .../views/EventSubscriber/RouteSubscriber.php |  1 +
 .../FilterBooleanOperatorStringTest.php       |  2 +-
 .../Handler/FilterBooleanOperatorTest.php     |  2 +-
 .../views/Tests/Handler/FilterDateTest.php    |  2 +-
 .../Tests/Handler/FilterEqualityTest.php      |  2 +-
 .../Tests/Handler/FilterInOperatorTest.php    |  2 +-
 .../views/Tests/Handler/FilterNumericTest.php |  2 +-
 .../views/Tests/Handler/FilterStringTest.php  |  2 +-
 .../views/Tests/Handler/HandlerTest.php       |  1 -
 .../Drupal/views/Tests/Plugin/AccessTest.php  |  4 +-
 .../Tests/Plugin/DisplayAttachmentTest.php    |  3 -
 .../views/Tests/Plugin/DisplayPageTest.php    | 11 ++-
 .../Drupal/views/Tests/Plugin/DisplayTest.php |  1 +
 .../views/Tests/Plugin/DisplayUnitTest.php    |  2 +-
 .../views/Tests/Plugin/ExposedFormTest.php    |  6 --
 .../views/Tests/Plugin/MiniPagerTest.php      |  1 -
 .../views/Tests/Plugin/RowEntityTest.php      |  2 +-
 .../Drupal/views/Tests/ViewUnitTestBase.php   |  3 +
 core/modules/views/views.module               |  9 +--
 .../views_ui/Tests/ExposedFormUITest.php      |  1 -
 .../Drupal/views_ui/Tests/SettingsTest.php    |  2 +-
 .../Tests/Core/Routing/NullRouteBuilder.php   | 18 +++++
 .../Tests/Core/Routing/RouteBuilderTest.php   | 52 +++++++++++++-
 50 files changed, 360 insertions(+), 116 deletions(-)
 create mode 100644 core/lib/Drupal/Core/EventSubscriber/RouterRebuildSubscriber.php
 create mode 100644 core/lib/Drupal/Core/Routing/RouteBuilderInterface.php
 create mode 100644 core/tests/Drupal/Tests/Core/Routing/NullRouteBuilder.php

diff --git a/core/core.services.yml b/core/core.services.yml
index 0bc1c53ee248..f00c4502e3f4 100644
--- a/core/core.services.yml
+++ b/core/core.services.yml
@@ -267,7 +267,9 @@ services:
       - [fromRequest, ['@request']]
   router.route_provider:
     class: Drupal\Core\Routing\RouteProvider
-    arguments: ['@database']
+    arguments: ['@database', '@router.builder']
+    tags:
+      - { name: event_subscriber }
   router.matcher.final_matcher:
     class: Drupal\Core\Routing\UrlMatcher
   router.matcher:
@@ -310,7 +312,12 @@ services:
     arguments: ['@database']
   router.builder:
     class: Drupal\Core\Routing\RouteBuilder
-    arguments: ['@router.dumper', '@lock', '@event_dispatcher', '@module_handler', '@controller_resolver']
+    arguments: ['@router.dumper', '@lock', '@event_dispatcher', '@module_handler', '@controller_resolver', '@state']
+  router.rebuild_subscriber:
+    class: Drupal\Core\EventSubscriber\RouterRebuildSubscriber
+    arguments: ['@router.builder']
+    tags:
+      - { name: event_subscriber }
   path.alias_manager.cached:
     class: Drupal\Core\CacheDecorator\AliasManagerCacheDecorator
     arguments: ['@path.alias_manager', '@cache.path']
diff --git a/core/includes/common.inc b/core/includes/common.inc
index 8eec0acf0641..580ef214ad5d 100644
--- a/core/includes/common.inc
+++ b/core/includes/common.inc
@@ -4889,7 +4889,6 @@ function drupal_flush_all_caches() {
   // Important: This rebuild must happen last, so the menu router is guaranteed
   // to be based on up to date information.
   \Drupal::service('router.builder')->rebuild();
-  menu_router_rebuild();
 
   // Re-initialize the maintenance theme, if the current request attempted to
   // use it. Unlike regular usages of this function, the installer and update
diff --git a/core/includes/menu.inc b/core/includes/menu.inc
index 9d0d7474dba7..d54fe068cf1a 100644
--- a/core/includes/menu.inc
+++ b/core/includes/menu.inc
@@ -475,13 +475,7 @@ function menu_get_item($path = NULL, $router_item = NULL) {
     $router_items[$path] = $router_item;
   }
   if (!isset($router_items[$path])) {
-    // Rebuild if we know it's needed, or if the menu masks are missing which
-    // occurs rarely, likely due to a race condition of multiple rebuilds.
-    if (\Drupal::state()->get('menu_rebuild_needed') || !\Drupal::state()->get('menu.masks')) {
-      menu_router_rebuild();
-      \Drupal::service('router.builder')->rebuild();
-      Cache::deleteTags(array('local_task' => 1));
-    }
+    \Drupal::service('router.builder')->rebuildIfNeeded();
     $original_map = arg(NULL, $path);
 
     $parts = array_slice($original_map, 0, MENU_MAX_PARTS);
@@ -2536,13 +2530,13 @@ function menu_router_rebuild() {
   $transaction = db_transaction();
 
   try {
+    // Ensure the route based router is up to date.
+    \Drupal::service('router.builder')->rebuildIfNeeded();
     list($menu) = menu_router_build(TRUE);
     _menu_navigation_links_rebuild($menu);
     // Clear the menu, page and block caches.
     menu_cache_clear_all();
     _menu_clear_page_cache();
-    // Indicate that the menu has been successfully rebuilt.
-    \Drupal::state()->delete('menu_rebuild_needed');
   }
   catch (Exception $e) {
     $transaction->rollback();
@@ -2744,6 +2738,9 @@ function _menu_navigation_links_rebuild($menu) {
   }
 
   // Find any item whose router path does not exist any more.
+  // Do not delete entries with an empty path as this can remove menu links in
+  // the process of being created.
+  $paths[] = '';
   $query = \Drupal::entityQuery('menu_link')
     ->condition('router_path', $paths, 'NOT IN')
     ->condition('external', 0)
diff --git a/core/lib/Drupal/Core/EventSubscriber/RouterRebuildSubscriber.php b/core/lib/Drupal/Core/EventSubscriber/RouterRebuildSubscriber.php
new file mode 100644
index 000000000000..d432656bf05a
--- /dev/null
+++ b/core/lib/Drupal/Core/EventSubscriber/RouterRebuildSubscriber.php
@@ -0,0 +1,71 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\EventSubscriber\RouterRebuildSubscriber.
+ */
+
+namespace Drupal\Core\EventSubscriber;
+
+use Drupal\Core\Cache\Cache;
+use Drupal\Core\Routing\RouteBuilderInterface;
+use Drupal\Core\Routing\RoutingEvents;
+use Symfony\Component\EventDispatcher\Event;
+use Symfony\Component\EventDispatcher\EventSubscriberInterface;
+use Symfony\Component\HttpKernel\Event\PostResponseEvent;
+use Symfony\Component\HttpKernel\KernelEvents;
+
+/**
+ * Rebuilds the router and menu_router if necessary.
+ */
+class RouterRebuildSubscriber implements EventSubscriberInterface {
+
+  /**
+   * @var \Drupal\Core\Routing\RouteBuilderInterface
+   */
+  protected $routeBuilder;
+
+  /**
+   * Constructs the RouterRebuildSubscriber object.
+   *
+   * @param \Drupal\Core\Routing\RouteBuilderInterface $route_builder
+   *   The route builder.
+   */
+  public function __construct(RouteBuilderInterface $route_builder) {
+    $this->routeBuilder = $route_builder;
+  }
+
+  /**
+   * Rebuilds routers if necessary.
+   *
+   * @param \Symfony\Component\HttpKernel\Event\PostResponseEvent $event
+   *   The event object.
+   */
+  public function onKernelTerminate(PostResponseEvent $event) {
+    $this->routeBuilder->rebuildIfNeeded();
+  }
+
+  /**
+   * Rebuilds the menu_router and deletes the local_task cache tag.
+   *
+   * @param \Symfony\Component\EventDispatcher\Event $event
+   *   The event object.
+   */
+  public function onRouterRebuild(Event $event) {
+    menu_router_rebuild();
+    Cache::deleteTags(array('local_task' => 1));
+  }
+
+  /**
+   * Registers the methods in this class that should be listeners.
+   *
+   * @return array
+   *   An array of event listener definitions.
+   */
+  static function getSubscribedEvents() {
+    $events[KernelEvents::TERMINATE][] = array('onKernelTerminate', 200);
+    $events[RoutingEvents::FINISHED][] = array('onRouterRebuild', 200);
+    return $events;
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Extension/ThemeHandler.php b/core/lib/Drupal/Core/Extension/ThemeHandler.php
index c25dcd962752..7c71eba93a73 100644
--- a/core/lib/Drupal/Core/Extension/ThemeHandler.php
+++ b/core/lib/Drupal/Core/Extension/ThemeHandler.php
@@ -84,7 +84,7 @@ class ThemeHandler implements ThemeHandlerInterface {
    *
    * @var \Drupal\Core\Routing\RouteBuilder
    */
-  protected $routerBuilder;
+  protected $routeBuilder;
 
   /**
    * The system listing info
@@ -460,7 +460,7 @@ protected function getSystemListingInfo() {
    */
   protected function resetSystem() {
     if ($this->routeBuilder) {
-      $this->routeBuilder->rebuild();
+      $this->routeBuilder->setRebuildNeeded();
     }
     $this->systemListReset();
 
diff --git a/core/lib/Drupal/Core/Routing/RouteBuilder.php b/core/lib/Drupal/Core/Routing/RouteBuilder.php
index d953fddff47a..eebe414b88fa 100644
--- a/core/lib/Drupal/Core/Routing/RouteBuilder.php
+++ b/core/lib/Drupal/Core/Routing/RouteBuilder.php
@@ -9,6 +9,7 @@
 
 use Drupal\Component\Discovery\YamlDiscovery;
 use Drupal\Core\Controller\ControllerResolverInterface;
+use Drupal\Core\KeyValueStore\StateInterface;
 use Symfony\Component\EventDispatcher\Event;
 use Symfony\Component\EventDispatcher\EventDispatcherInterface;
 use Symfony\Component\Routing\RouteCollection;
@@ -23,7 +24,7 @@
  * Because this class makes use of the modules system, it cannot currently
  * be unit tested.
  */
-class RouteBuilder {
+class RouteBuilder implements RouteBuilderInterface {
 
   /**
    * The dumper to which we should send collected routes.
@@ -68,7 +69,7 @@ class RouteBuilder {
   protected $controllerResolver;
 
   /**
-   * Construcs the RouteBuilder using the passed MatcherDumperInterface.
+   * Constructs the RouteBuilder using the passed MatcherDumperInterface.
    *
    * @param \Drupal\Core\Routing\MatcherDumperInterface $dumper
    *   The matcher dumper used to store the route information.
@@ -81,19 +82,17 @@ class RouteBuilder {
    * @param \Drupal\Core\Controller\ControllerResolverInterface $controller_resolver
    *   The controller resolver.
    */
-  public function __construct(MatcherDumperInterface $dumper, LockBackendInterface $lock, EventDispatcherInterface $dispatcher, ModuleHandlerInterface $module_handler, ControllerResolverInterface $controller_resolver) {
+  public function __construct(MatcherDumperInterface $dumper, LockBackendInterface $lock, EventDispatcherInterface $dispatcher, ModuleHandlerInterface $module_handler, ControllerResolverInterface $controller_resolver, StateInterface $state = NULL) {
     $this->dumper = $dumper;
     $this->lock = $lock;
     $this->dispatcher = $dispatcher;
     $this->moduleHandler = $module_handler;
     $this->controllerResolver = $controller_resolver;
+    $this->state = $state;
   }
 
   /**
-   * Rebuilds the route info and dumps to dumper.
-   *
-   * @return bool
-   *   Returns TRUE if the rebuild succeeds, FALSE otherwise.
+   * {@inheritdoc}
    */
   public function rebuild() {
     if (!$this->lock->acquire('router_rebuild')) {
@@ -157,11 +156,29 @@ public function rebuild() {
     $this->dumper->addRoutes($collection);
     $this->dumper->dump(array('provider' => 'dynamic_routes'));
 
+    $this->state->delete(static::REBUILD_NEEDED);
     $this->lock->release('router_rebuild');
     $this->dispatcher->dispatch(RoutingEvents::FINISHED, new Event());
     return TRUE;
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function rebuildIfNeeded() {
+    if ($this->state->get(static::REBUILD_NEEDED, FALSE)) {
+      return $this->rebuild();
+    }
+    return FALSE;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function setRebuildNeeded() {
+    $this->state->set(static::REBUILD_NEEDED, TRUE);
+  }
+
   /**
    * Returns the YAML discovery for getting all the .routing.yml files.
    *
diff --git a/core/lib/Drupal/Core/Routing/RouteBuilderInterface.php b/core/lib/Drupal/Core/Routing/RouteBuilderInterface.php
new file mode 100644
index 000000000000..2c9ef24c18ad
--- /dev/null
+++ b/core/lib/Drupal/Core/Routing/RouteBuilderInterface.php
@@ -0,0 +1,35 @@
+<?php
+
+/**
+ * @file
+ * Definition of Drupal\Core\Routing\RouteBuilderInterface.
+ */
+
+namespace Drupal\Core\Routing;
+
+interface RouteBuilderInterface {
+
+  const REBUILD_NEEDED = 'router_rebuild_needed';
+
+  /**
+   * Rebuilds the route info and dumps to dumper.
+   *
+   * @return bool
+   *   Returns TRUE if the rebuild succeeds, FALSE otherwise.
+   */
+  public function rebuild();
+
+  /**
+   * Rebuilds the route info and dumps to dumper if necessary.
+   *
+   * @return bool
+   *   Returns TRUE if the rebuild occurs, FALSE otherwise.
+   */
+  public function rebuildIfNeeded();
+
+  /**
+   * Sets the router to be rebuilt next time rebuildIfNeeded() is called.
+   */
+  public function setRebuildNeeded();
+
+}
diff --git a/core/lib/Drupal/Core/Routing/RouteBuilderStatic.php b/core/lib/Drupal/Core/Routing/RouteBuilderStatic.php
index d6161f602999..327621eb0e8d 100644
--- a/core/lib/Drupal/Core/Routing/RouteBuilderStatic.php
+++ b/core/lib/Drupal/Core/Routing/RouteBuilderStatic.php
@@ -10,14 +10,39 @@
 /**
  * This builds a static version of the router.
  */
-class RouteBuilderStatic {
+class RouteBuilderStatic implements RouteBuilderInterface {
 
   /**
-   * Rebuilds router.
+   * Marks a rebuild as being necessary.
+   *
+   * @var bool
+   */
+  protected $rebuildNeeded = FALSE;
+
+  /**
+   * @inheritdoc
    */
   public function rebuild() {
     // @todo Add the route for the batch pages when that conversion happens,
     //   http://drupal.org/node/1987816.
   }
 
+  /**
+   * @inheritdoc
+   */
+  public function rebuildIfNeeded(){
+    if ($this->rebuildNeeded && $this->rebuild()) {
+      $this->rebuildNeeded = FALSE;
+      return TRUE;
+    }
+    return FALSE;
+  }
+
+  /**
+   * @inheritdoc
+   */
+  public function setRebuildNeeded() {
+    $this->rebuildNeeded = TRUE;
+  }
+
 }
diff --git a/core/lib/Drupal/Core/Routing/RouteProvider.php b/core/lib/Drupal/Core/Routing/RouteProvider.php
index 18dc50621330..20b0ec138706 100644
--- a/core/lib/Drupal/Core/Routing/RouteProvider.php
+++ b/core/lib/Drupal/Core/Routing/RouteProvider.php
@@ -8,6 +8,7 @@
 namespace Drupal\Core\Routing;
 
 use Drupal\Component\Utility\String;
+use Symfony\Component\EventDispatcher\EventSubscriberInterface;
 use Symfony\Component\HttpFoundation\Request;
 use Symfony\Component\Routing\Exception\RouteNotFoundException;
 use Symfony\Component\Routing\RouteCollection;
@@ -18,7 +19,7 @@
 /**
  * A Route Provider front-end for all Drupal-stored routes.
  */
-class RouteProvider implements RouteProviderInterface {
+class RouteProvider implements RouteProviderInterface, EventSubscriberInterface {
 
   /**
    * The database connection from which to read route information.
@@ -34,6 +35,13 @@ class RouteProvider implements RouteProviderInterface {
    */
   protected $tableName;
 
+  /**
+   * The route builder.
+   *
+   * @var \Drupal\Core\Routing\RouteBuilderInterface
+   */
+  protected $routeBuilder;
+
   /**
    * A cache of already-loaded routes, keyed by route name.
    *
@@ -46,11 +54,14 @@ class RouteProvider implements RouteProviderInterface {
    *
    * @param \Drupal\Core\Database\Connection $connection
    *   A database connection object.
+   * @param \Drupal\Core\Routing\RouteBuilderInterface $route_builder
+   *   The route builder.
    * @param string $table
    *   The table in the database to use for matching.
    */
-  public function __construct(Connection $connection, $table = 'router') {
+  public function __construct(Connection $connection, RouteBuilderInterface $route_builder, $table = 'router') {
     $this->connection = $connection;
+    $this->routeBuilder = $route_builder;
     $this->tableName = $table;
   }
 
@@ -99,7 +110,12 @@ public function getRouteCollectionForRequest(Request $request) {
 
     $collection = $this->getRoutesByPath($path);
 
-    if (!count($collection)) {
+    // Try rebuilding the router if it is necessary.
+    if (!$collection->count() && $this->routeBuilder->rebuildIfNeeded()) {
+      $collection = $this->getRoutesByPath($path);
+    }
+
+    if (!$collection->count()) {
       throw new ResourceNotFoundException(String::format("The route for '@path' could not be found", array('@path' => $path)));
     }
 
@@ -269,4 +285,19 @@ public function getAllRoutes() {
     return new LazyLoadingRouteCollection($this->connection, $this->tableName);
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function reset() {
+    $this->routes  = array();
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  static function getSubscribedEvents() {
+    $events[RoutingEvents::FINISHED][] = array('reset');
+    return $events;
+  }
+
 }
diff --git a/core/lib/Drupal/Core/Routing/RouteProviderInterface.php b/core/lib/Drupal/Core/Routing/RouteProviderInterface.php
index f84a19ba26e1..8c8e3561d4fd 100644
--- a/core/lib/Drupal/Core/Routing/RouteProviderInterface.php
+++ b/core/lib/Drupal/Core/Routing/RouteProviderInterface.php
@@ -40,4 +40,9 @@ public function getRoutesByPattern($pattern);
    */
   public function getAllRoutes();
 
+  /**
+   * Resets the route provider object.
+   */
+  public function reset();
+
 }
diff --git a/core/modules/block/lib/Drupal/block/Tests/BlockHiddenRegionTest.php b/core/modules/block/lib/Drupal/block/Tests/BlockHiddenRegionTest.php
index 208710e72f00..3fa85d632589 100644
--- a/core/modules/block/lib/Drupal/block/Tests/BlockHiddenRegionTest.php
+++ b/core/modules/block/lib/Drupal/block/Tests/BlockHiddenRegionTest.php
@@ -64,8 +64,9 @@ public function testBlockNotInHiddenRegion() {
     \Drupal::config('system.theme')
       ->set('default', $theme)
       ->save();
-    \Drupal::service('router.builder')->rebuild();
-    menu_router_rebuild();
+    // Enabling a theme will cause the kernel terminate event to rebuild the
+    // router. Simulate that here.
+    \Drupal::service('router.builder')->rebuildIfNeeded();
 
     // Ensure that "block_test_theme" is set as the default theme.
     $this->drupalGet('admin/structure/block');
diff --git a/core/modules/color/lib/Drupal/color/Tests/ColorTest.php b/core/modules/color/lib/Drupal/color/Tests/ColorTest.php
index 419315288740..a22b003881a0 100644
--- a/core/modules/color/lib/Drupal/color/Tests/ColorTest.php
+++ b/core/modules/color/lib/Drupal/color/Tests/ColorTest.php
@@ -48,8 +48,6 @@ function setUp() {
       ),
     );
     theme_enable(array_keys($this->themes));
-    $this->container->get('router.builder')->rebuild();
-    menu_router_rebuild();
 
     // Array filled with valid and not valid color values
     $this->colorTests = array(
diff --git a/core/modules/config_translation/lib/Drupal/config_translation/Tests/ConfigTranslationUiTest.php b/core/modules/config_translation/lib/Drupal/config_translation/Tests/ConfigTranslationUiTest.php
index c6e40943300b..393db3332796 100644
--- a/core/modules/config_translation/lib/Drupal/config_translation/Tests/ConfigTranslationUiTest.php
+++ b/core/modules/config_translation/lib/Drupal/config_translation/Tests/ConfigTranslationUiTest.php
@@ -669,8 +669,9 @@ public function testThemeDiscovery() {
     // Enable the test theme and rebuild routes.
     $theme = 'config_translation_test_theme';
     theme_enable(array($theme));
-    \Drupal::service('router.builder')->rebuild();
-    menu_router_rebuild();
+    // Enabling a theme will cause the kernel terminate event to rebuild the
+    // router. Simulate that here.
+    \Drupal::service('router.builder')->rebuildIfNeeded();
 
     $this->drupalLogin($this->admin_user);
 
diff --git a/core/modules/field_ui/field_ui.module b/core/modules/field_ui/field_ui.module
index 9461296b05dd..5751c15f19bb 100644
--- a/core/modules/field_ui/field_ui.module
+++ b/core/modules/field_ui/field_ui.module
@@ -130,7 +130,7 @@ function field_ui_entity_info(&$entity_info) {
 function field_ui_entity_bundle_create($entity_type, $bundle) {
   // When a new bundle is created, the menu needs to be rebuilt to add our
   // menu item tabs.
-  \Drupal::state()->set('menu_rebuild_needed', TRUE);
+  \Drupal::service('router.builder')->setRebuildNeeded();
 }
 
 /**
@@ -139,7 +139,7 @@ function field_ui_entity_bundle_create($entity_type, $bundle) {
 function field_ui_entity_bundle_rename($entity_type, $bundle_old, $bundle_new) {
   // When a bundle is renamed, the menu needs to be rebuilt to add our
   // menu item tabs.
-  \Drupal::state()->set('menu_rebuild_needed', TRUE);
+  \Drupal::service('router.builder')->setRebuildNeeded();
 }
 
 /**
@@ -237,14 +237,14 @@ function field_ui_library_info() {
  * Implements hook_view_mode_presave().
  */
 function field_ui_view_mode_presave(EntityViewModeInterface $view_mode) {
-  \Drupal::state()->set('menu_rebuild_needed', TRUE);
+  \Drupal::service('router.builder')->setRebuildNeeded();
 }
 
 /**
  * Implements hook_view_mode_delete().
  */
 function field_ui_view_mode_delete(EntityViewModeInterface $view_mode) {
-  \Drupal::state()->set('menu_rebuild_needed', TRUE);
+  \Drupal::service('router.builder')->setRebuildNeeded();
 }
 
 /**
diff --git a/core/modules/file/lib/Drupal/file/Tests/Views/ExtensionViewsFieldTest.php b/core/modules/file/lib/Drupal/file/Tests/Views/ExtensionViewsFieldTest.php
index 87ab5f89c10d..9683c1fa3f4b 100644
--- a/core/modules/file/lib/Drupal/file/Tests/Views/ExtensionViewsFieldTest.php
+++ b/core/modules/file/lib/Drupal/file/Tests/Views/ExtensionViewsFieldTest.php
@@ -18,7 +18,7 @@ class ExtensionViewsFieldTest extends ViewUnitTestBase {
   /**
    * {@inheritdoc}
    */
-  public static $modules = array('file', 'file_test_views');
+  public static $modules = array('file', 'file_test_views', 'user');
 
   /**
    * Views used by this test.
diff --git a/core/modules/menu/menu.install b/core/modules/menu/menu.install
index b7dc67d65435..2a61238f1d19 100644
--- a/core/modules/menu/menu.install
+++ b/core/modules/menu/menu.install
@@ -13,7 +13,6 @@
 function menu_install() {
   // Add a link for each custom menu.
   \Drupal::service('router.builder')->rebuild();
-  menu_router_rebuild();
   $system_link = entity_load_multiple_by_properties('menu_link', array('link_path' => 'admin/structure/menu', 'module' => 'system'));
   $system_link = reset($system_link);
 
diff --git a/core/modules/menu_link/lib/Drupal/menu_link/Entity/MenuLink.php b/core/modules/menu_link/lib/Drupal/menu_link/Entity/MenuLink.php
index ba0397e82ff5..cf9eeccd4403 100644
--- a/core/modules/menu_link/lib/Drupal/menu_link/Entity/MenuLink.php
+++ b/core/modules/menu_link/lib/Drupal/menu_link/Entity/MenuLink.php
@@ -539,7 +539,7 @@ public function preSave(EntityStorageControllerInterface $storage_controller) {
     }
     // Find the route_name.
     if (!isset($this->route_name)) {
-      if ($result = \Drupal::service('router.matcher.final_matcher')->findRouteNameParameters($this->link_path)) {
+      if (!$this->external && $result = \Drupal::service('router.matcher.final_matcher')->findRouteNameParameters($this->link_path)) {
         list($this->route_name, $this->route_parameters) = $result;
       }
       else {
diff --git a/core/modules/menu_link/lib/Drupal/menu_link/MenuLinkStorageController.php b/core/modules/menu_link/lib/Drupal/menu_link/MenuLinkStorageController.php
index 0e95f42353bb..61b4b6c0da87 100644
--- a/core/modules/menu_link/lib/Drupal/menu_link/MenuLinkStorageController.php
+++ b/core/modules/menu_link/lib/Drupal/menu_link/MenuLinkStorageController.php
@@ -121,7 +121,7 @@ public function save(EntityInterface $entity) {
       }
 
       if ($entity->isNew()) {
-        $entity->mlid = $this->database->insert($this->entityInfo->getBaseTable())->fields(array('menu_name' => 'tools'))->execute();
+        $entity->mlid = $this->database->insert($this->entityInfo->getBaseTable())->fields(array('menu_name' => $entity->menu_name))->execute();
         $entity->enforceIsNew();
       }
 
diff --git a/core/modules/search/lib/Drupal/search/Entity/SearchPage.php b/core/modules/search/lib/Drupal/search/Entity/SearchPage.php
index d0419d46c2ea..d40900d1c069 100644
--- a/core/modules/search/lib/Drupal/search/Entity/SearchPage.php
+++ b/core/modules/search/lib/Drupal/search/Entity/SearchPage.php
@@ -205,11 +205,7 @@ public function preSave(EntityStorageControllerInterface $storage_controller) {
    */
   public function postSave(EntityStorageControllerInterface $storage_controller, $update = TRUE) {
     parent::postSave($storage_controller, $update);
-
-    $this->state()->set('menu_rebuild_needed', TRUE);
-    // @todo The above call should be sufficient, but it is not until
-    //   https://drupal.org/node/2167323 is fixed.
-    \Drupal::service('router.builder')->rebuild();
+    $this->routeBuilder()->setRebuildNeeded();
   }
 
   /**
@@ -239,13 +235,13 @@ public static function sort($a, $b) {
   }
 
   /**
-   * Wraps the state storage.
+   * Wraps the route builder.
    *
-   * @return \Drupal\Core\KeyValueStore\StateInterface
+   * @return \Drupal\Core\Routing\RouteBuilderInterface
    *   An object for state storage.
    */
-  protected function state() {
-    return \Drupal::state();
+  protected function routeBuilder() {
+    return \Drupal::service('router.builder');
   }
 
   /**
diff --git a/core/modules/search/lib/Drupal/search/Tests/SearchKeywordsConditionsTest.php b/core/modules/search/lib/Drupal/search/Tests/SearchKeywordsConditionsTest.php
index 8e5f7e0d0c0e..8b1ca49011f5 100644
--- a/core/modules/search/lib/Drupal/search/Tests/SearchKeywordsConditionsTest.php
+++ b/core/modules/search/lib/Drupal/search/Tests/SearchKeywordsConditionsTest.php
@@ -34,8 +34,6 @@ function setUp() {
     $this->searching_user = $this->drupalCreateUser(array('search content', 'access content', 'access comments', 'skip comment approval'));
     // Login with sufficient privileges.
     $this->drupalLogin($this->searching_user);
-    // Test with all search modules enabled.
-    \Drupal::state()->set('menu_rebuild_needed', TRUE);
   }
 
   /**
diff --git a/core/modules/search/lib/Drupal/search/Tests/SearchPageOverrideTest.php b/core/modules/search/lib/Drupal/search/Tests/SearchPageOverrideTest.php
index e26b2bd1d7e5..88d9c54be329 100644
--- a/core/modules/search/lib/Drupal/search/Tests/SearchPageOverrideTest.php
+++ b/core/modules/search/lib/Drupal/search/Tests/SearchPageOverrideTest.php
@@ -35,13 +35,9 @@ function setUp() {
     // Login as a user that can create and search content.
     $this->search_user = $this->drupalCreateUser(array('search content', 'administer search'));
     $this->drupalLogin($this->search_user);
-
-    // Enable the extra type module for searching.
-    \Drupal::service('router.builder')->rebuild();
   }
 
   function testSearchPageHook() {
-    $this->container->get('router.builder')->rebuild();
     $keys = 'bike shed ' . $this->randomName();
     $this->drupalGet("search/dummy_path/{$keys}");
     $this->assertText('Dummy search snippet', 'Dummy search snippet is shown');
diff --git a/core/modules/system/lib/Drupal/system/Entity/Menu.php b/core/modules/system/lib/Drupal/system/Entity/Menu.php
index fb09c0464e4b..644629abc644 100644
--- a/core/modules/system/lib/Drupal/system/Entity/Menu.php
+++ b/core/modules/system/lib/Drupal/system/Entity/Menu.php
@@ -8,6 +8,7 @@
 namespace Drupal\system\Entity;
 
 use Drupal\Core\Config\Entity\ConfigEntityBase;
+use Drupal\Core\Entity\EntityStorageControllerInterface;
 use Drupal\system\MenuInterface;
 
 /**
diff --git a/core/modules/system/lib/Drupal/system/Tests/Menu/MenuRouterTest.php b/core/modules/system/lib/Drupal/system/Tests/Menu/MenuRouterTest.php
index ccc327dcba38..6d0fee8eaf34 100644
--- a/core/modules/system/lib/Drupal/system/Tests/Menu/MenuRouterTest.php
+++ b/core/modules/system/lib/Drupal/system/Tests/Menu/MenuRouterTest.php
@@ -186,7 +186,6 @@ protected function doTestMenuName() {
     // rebuild.
     menu_test_menu_name('changed');
     \Drupal::service('router.builder')->rebuild();
-    menu_router_rebuild();
 
     $menu_links = entity_load_multiple_by_properties('menu_link', array('router_path' => 'menu_name_test'));
     $menu_link = reset($menu_links);
diff --git a/core/modules/system/lib/Drupal/system/Tests/Menu/RebuildTest.php b/core/modules/system/lib/Drupal/system/Tests/Menu/RebuildTest.php
index 8c01748c0414..e02c9c4f23d4 100644
--- a/core/modules/system/lib/Drupal/system/Tests/Menu/RebuildTest.php
+++ b/core/modules/system/lib/Drupal/system/Tests/Menu/RebuildTest.php
@@ -7,10 +7,11 @@
 
 namespace Drupal\system\Tests\Menu;
 
+use Drupal\Core\Routing\RouteBuilderInterface;
 use Drupal\simpletest\WebTestBase;
 
 /**
- * Tests rebuilding the menu by setting 'menu_rebuild_needed.'
+ * Tests rebuilding the router.
  */
 class RebuildTest extends WebTestBase {
   public static function getInfo() {
@@ -22,9 +23,9 @@ public static function getInfo() {
   }
 
   /**
-   * Test if the 'menu_rebuild_needed' variable triggers a menu_rebuild() call.
+   * Tests that set a router rebuild needed works.
    */
-  function testMenuRebuildByVariable() {
+  function testMenuRebuild() {
     // Check if 'admin' path exists.
     $admin_exists = db_query('SELECT path from {menu_router} WHERE path = :path', array(':path' => 'admin'))->fetchField();
     $this->assertEqual($admin_exists, 'admin', "The path 'admin/' exists prior to deleting.");
@@ -36,9 +37,9 @@ function testMenuRebuildByVariable() {
     $admin_exists = db_query('SELECT path from {menu_router} WHERE path = :path', array(':path' => 'admin'))->fetchField();
     $this->assertFalse($admin_exists, "The path 'admin/' has been deleted and doesn't exist in the database.");
 
-    // Now we enable the rebuild variable and send a request to rebuild the menu
-    // item. Now 'admin' should exist.
-    \Drupal::state()->set('menu_rebuild_needed', TRUE);
+    // Now we set the router to be rebuilt. After the rebuild 'admin' should exist.
+    \Drupal::service('router.builder')->setRebuildNeeded();
+
     // The request should trigger the rebuild.
     $this->drupalGet('<front>');
     $admin_exists = db_query('SELECT path from {menu_router} WHERE path = :path', array(':path' => 'admin'))->fetchField();
diff --git a/core/modules/system/lib/Drupal/system/Tests/Routing/MockRouteProvider.php b/core/modules/system/lib/Drupal/system/Tests/Routing/MockRouteProvider.php
index de56a90d1781..4a6ece5b9ca2 100644
--- a/core/modules/system/lib/Drupal/system/Tests/Routing/MockRouteProvider.php
+++ b/core/modules/system/lib/Drupal/system/Tests/Routing/MockRouteProvider.php
@@ -82,4 +82,11 @@ public function getAllRoutes() {
     return $this->routes->all();
   }
 
+  /**
+   * @inheritdoc}
+   */
+  public function reset() {
+    $this->routes = array();
+  }
+
 }
diff --git a/core/modules/system/lib/Drupal/system/Tests/Routing/RouteProviderTest.php b/core/modules/system/lib/Drupal/system/Tests/Routing/RouteProviderTest.php
index bae0909a4a3b..b1437978c7ad 100644
--- a/core/modules/system/lib/Drupal/system/Tests/Routing/RouteProviderTest.php
+++ b/core/modules/system/lib/Drupal/system/Tests/Routing/RouteProviderTest.php
@@ -18,6 +18,7 @@
 use Drupal\Core\Database\Database;
 use Drupal\Core\Routing\MatcherDumper;
 use Drupal\Tests\Core\Routing\RoutingFixtures;
+use Drupal\Tests\Core\Routing\NullRouteBuilder;
 
 /**
  * Basic tests for the RouteProvider.
@@ -31,6 +32,13 @@ class RouteProviderTest extends UnitTestBase {
    */
   protected $fixtures;
 
+  /**
+   * A null route builder to enable testing of the route provider.
+   *
+   * @var \Drupal\Core\Routing\RouteBuilderInterface
+   */
+  protected $routeBuilder;
+
   public static function getInfo() {
     return array(
       'name' => 'Route Provider tests',
@@ -43,6 +51,7 @@ function __construct($test_id = NULL) {
     parent::__construct($test_id);
 
     $this->fixtures = new RoutingFixtures();
+    $this->routeBuilder = new NullRouteBuilder();
   }
 
   public function tearDown() {
@@ -57,7 +66,7 @@ public function tearDown() {
   public function testCandidateOutlines() {
 
     $connection = Database::getConnection();
-    $provider = new RouteProvider($connection);
+    $provider = new RouteProvider($connection, $this->routeBuilder);
 
     $parts = array('node', '5', 'edit');
 
@@ -81,7 +90,7 @@ public function testCandidateOutlines() {
    */
   function testExactPathMatch() {
     $connection = Database::getConnection();
-    $provider = new RouteProvider($connection, 'test_routes');
+    $provider = new RouteProvider($connection, $this->routeBuilder, 'test_routes');
 
     $this->fixtures->createTables($connection);
 
@@ -105,7 +114,7 @@ function testExactPathMatch() {
    */
   function testOutlinePathMatch() {
     $connection = Database::getConnection();
-    $provider = new RouteProvider($connection, 'test_routes');
+    $provider = new RouteProvider($connection, $this->routeBuilder, 'test_routes');
 
     $this->fixtures->createTables($connection);
 
@@ -134,7 +143,7 @@ function testOutlinePathMatch() {
    */
   function testOutlinePathMatchTrailingSlash() {
     $connection = Database::getConnection();
-    $provider = new RouteProvider($connection, 'test_routes');
+    $provider = new RouteProvider($connection, $this->routeBuilder, 'test_routes');
 
     $this->fixtures->createTables($connection);
 
@@ -163,7 +172,7 @@ function testOutlinePathMatchTrailingSlash() {
    */
   function testOutlinePathMatchDefaults() {
     $connection = Database::getConnection();
-    $provider = new RouteProvider($connection, 'test_routes');
+    $provider = new RouteProvider($connection, $this->routeBuilder, 'test_routes');
 
     $this->fixtures->createTables($connection);
 
@@ -201,7 +210,7 @@ function testOutlinePathMatchDefaults() {
    */
   function testOutlinePathMatchDefaultsCollision() {
     $connection = Database::getConnection();
-    $provider = new RouteProvider($connection, 'test_routes');
+    $provider = new RouteProvider($connection, $this->routeBuilder, 'test_routes');
 
     $this->fixtures->createTables($connection);
 
@@ -240,7 +249,7 @@ function testOutlinePathMatchDefaultsCollision() {
    */
   function testOutlinePathMatchDefaultsCollision2() {
     $connection = Database::getConnection();
-    $provider = new RouteProvider($connection, 'test_routes');
+    $provider = new RouteProvider($connection, $this->routeBuilder, 'test_routes');
 
     $this->fixtures->createTables($connection);
 
@@ -279,7 +288,7 @@ function testOutlinePathMatchDefaultsCollision2() {
    */
   public function testOutlinePathMatchZero() {
     $connection = Database::getConnection();
-    $provider = new RouteProvider($connection, 'test_routes');
+    $provider = new RouteProvider($connection, $this->routeBuilder, 'test_routes');
 
     $this->fixtures->createTables($connection);
 
@@ -314,7 +323,7 @@ public function testOutlinePathMatchZero() {
    */
   function testOutlinePathNoMatch() {
     $connection = Database::getConnection();
-    $provider = new RouteProvider($connection, 'test_routes');
+    $provider = new RouteProvider($connection, $this->routeBuilder, 'test_routes');
 
     $this->fixtures->createTables($connection);
 
@@ -343,7 +352,7 @@ function testOutlinePathNoMatch() {
    */
   function testSystemPathMatch() {
     $connection = Database::getConnection();
-    $provider = new RouteProvider($connection, 'test_routes');
+    $provider = new RouteProvider($connection, $this->routeBuilder, 'test_routes');
 
     $this->fixtures->createTables($connection);
 
@@ -368,7 +377,7 @@ function testSystemPathMatch() {
    */
   protected function testRouteByName() {
     $connection = Database::getConnection();
-    $provider = new RouteProvider($connection, 'test_routes');
+    $provider = new RouteProvider($connection, $this->routeBuilder, 'test_routes');
 
     $this->fixtures->createTables($connection);
 
@@ -403,7 +412,7 @@ protected function testRouteByName() {
    */
   public function testGetRoutesByPatternWithLongPatterns() {
     $connection = Database::getConnection();
-    $provider = new RouteProvider($connection, 'test_routes');
+    $provider = new RouteProvider($connection, $this->routeBuilder, 'test_routes');
 
     $this->fixtures->createTables($connection);
 
diff --git a/core/modules/system/system.admin.inc b/core/modules/system/system.admin.inc
index d4c222d58ad4..3d14d09d2960 100644
--- a/core/modules/system/system.admin.inc
+++ b/core/modules/system/system.admin.inc
@@ -36,9 +36,7 @@ function system_theme_default() {
       // implementations, and saving the configuration before the theme_enable()
       // could result in a race condition where the theme is default but not
       // enabled.
-      \Drupal::service('router.builder')->rebuild();
-      menu_router_rebuild();
-      Cache::deleteTags(array('local_task' => TRUE));
+      \Drupal::service('router.builder')->setRebuildNeeded();
 
       // The status message depends on whether an admin theme is currently in use:
       // a value of 0 means the admin theme is set to be the default theme.
diff --git a/core/modules/views/lib/Drupal/views/EventSubscriber/RouteSubscriber.php b/core/modules/views/lib/Drupal/views/EventSubscriber/RouteSubscriber.php
index 9540364dc064..4188c474e2e1 100644
--- a/core/modules/views/lib/Drupal/views/EventSubscriber/RouteSubscriber.php
+++ b/core/modules/views/lib/Drupal/views/EventSubscriber/RouteSubscriber.php
@@ -181,6 +181,7 @@ protected function alterRoutes(RouteCollection $collection, $provider) {
    * {@inheritdoc}
    */
   public function routeRebuildFinished() {
+    $this->reset();
     $this->state->set('views.view_route_names', $this->viewRouteNames);
   }
 
diff --git a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterBooleanOperatorStringTest.php b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterBooleanOperatorStringTest.php
index 3617409784ba..d3c386242c70 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterBooleanOperatorStringTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterBooleanOperatorStringTest.php
@@ -49,7 +49,7 @@ public static function getInfo() {
   protected function setUp() {
     parent::setUp();
 
-    $this->installSchema('system', array('router', 'menu_router', 'variable', 'key_value_expire'));
+    $this->installSchema('system', array('menu_router', 'variable', 'key_value_expire'));
   }
 
 
diff --git a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterBooleanOperatorTest.php b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterBooleanOperatorTest.php
index 8ff7823d83fc..77841f37fd15 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterBooleanOperatorTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterBooleanOperatorTest.php
@@ -45,7 +45,7 @@ public static function getInfo() {
   protected function setUp() {
     parent::setUp();
 
-    $this->installSchema('system', array('router', 'menu_router', 'variable', 'key_value_expire'));
+    $this->installSchema('system', array('menu_router', 'variable', 'key_value_expire'));
   }
 
   /**
diff --git a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterDateTest.php b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterDateTest.php
index e73a5b6dc391..ead8b35c8ef9 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterDateTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterDateTest.php
@@ -151,7 +151,7 @@ protected function _testBetween() {
   protected function _testUiValidation() {
 
     $this->drupalLogin($this->drupalCreateUser(array('administer views', 'administer site configuration')));
-    menu_router_rebuild();
+
     $this->drupalGet('admin/structure/views/view/test_filter_date_between/edit');
     $this->drupalGet('admin/structure/views/nojs/handler/test_filter_date_between/default/filter/created');
 
diff --git a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterEqualityTest.php b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterEqualityTest.php
index 4ab42ad04d26..9b74533ef757 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterEqualityTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterEqualityTest.php
@@ -38,7 +38,7 @@ public static function getInfo() {
   protected function setUp() {
     parent::setUp();
 
-    $this->installSchema('system', array('router', 'menu_router', 'variable', 'key_value_expire'));
+    $this->installSchema('system', array('menu_router', 'variable', 'key_value_expire'));
   }
 
   function viewsData() {
diff --git a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterInOperatorTest.php b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterInOperatorTest.php
index 6e3e4e589ed5..0c1f32a9236b 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterInOperatorTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterInOperatorTest.php
@@ -39,7 +39,7 @@ public static function getInfo() {
   protected function setUp() {
     parent::setUp();
 
-    $this->installSchema('system', array('router', 'menu_router', 'variable', 'key_value_expire'));
+    $this->installSchema('system', array('menu_router', 'variable', 'key_value_expire'));
   }
 
   function viewsData() {
diff --git a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterNumericTest.php b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterNumericTest.php
index 62b16d30d070..a1f50ef18909 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterNumericTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterNumericTest.php
@@ -39,7 +39,7 @@ public static function getInfo() {
   protected function setUp() {
     parent::setUp();
 
-    $this->installSchema('system', array('router', 'menu_router', 'variable', 'key_value_expire'));
+    $this->installSchema('system', array('menu_router', 'variable', 'key_value_expire'));
   }
 
   function viewsData() {
diff --git a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterStringTest.php b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterStringTest.php
index cb444c923a86..f00eaf3ee061 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Handler/FilterStringTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Handler/FilterStringTest.php
@@ -38,7 +38,7 @@ public static function getInfo() {
   protected function setUp() {
     parent::setUp();
 
-    $this->installSchema('system', array('router', 'menu_router', 'variable', 'key_value_expire'));
+    $this->installSchema('system', array('menu_router', 'variable', 'key_value_expire'));
   }
 
   function viewsData() {
diff --git a/core/modules/views/lib/Drupal/views/Tests/Handler/HandlerTest.php b/core/modules/views/lib/Drupal/views/Tests/Handler/HandlerTest.php
index d4f7ba6bd35f..c719fce4f8a6 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Handler/HandlerTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Handler/HandlerTest.php
@@ -76,7 +76,6 @@ protected function viewsData() {
   function testFilterInOperatorUi() {
     $admin_user = $this->drupalCreateUser(array('administer views', 'administer site configuration'));
     $this->drupalLogin($admin_user);
-    menu_router_rebuild();
 
     $path = 'admin/structure/views/nojs/handler/test_filter_in_operator_ui/default/filter/type';
     $this->drupalGet($path);
diff --git a/core/modules/views/lib/Drupal/views/Tests/Plugin/AccessTest.php b/core/modules/views/lib/Drupal/views/Tests/Plugin/AccessTest.php
index 1532b40db885..9fe91bbe9af4 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Plugin/AccessTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Plugin/AccessTest.php
@@ -86,7 +86,9 @@ function testStaticAccessPlugin() {
     $display['display_options']['access']['options']['access'] = TRUE;
     $access_plugin->options['access'] = TRUE;
     $view->save();
-    $this->container->get('router.builder')->rebuild();
+    // Saving a view will cause the router to be rebuilt when the kernel
+    // termination event fires. Simulate that here.
+    $this->container->get('router.builder')->rebuildIfNeeded();
 
     $this->assertTrue($access_plugin->access($this->normal_user));
 
diff --git a/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayAttachmentTest.php b/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayAttachmentTest.php
index c5d6255a4b8f..7e550393adc0 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayAttachmentTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayAttachmentTest.php
@@ -40,9 +40,6 @@ protected function setUp() {
    * Tests the attachment plugin.
    */
   protected function testAttachment() {
-    // @todo Remove that once http://drupal.org/node/1828444 got in.
-    \Drupal::state()->set('menu_rebuild_needed', TRUE);
-
     $this->drupalGet('test-display-attachment');
 
     $result = $this->xpath('//div[contains(@class, "view-content")]');
diff --git a/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayPageTest.php b/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayPageTest.php
index 03862ade1eb2..5867c9823764 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayPageTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayPageTest.php
@@ -7,12 +7,9 @@
 
 namespace Drupal\views\Tests\Plugin;
 
-use Drupal\Core\Routing\RouteBuildEvent;
-use Drupal\views\EventSubscriber\RouteSubscriber;
 use Drupal\views\Tests\ViewUnitTestBase;
 use Symfony\Component\HttpFoundation\Request;
 use Symfony\Component\HttpKernel\HttpKernelInterface;
-use Symfony\Component\Routing\RouteCollection;
 
 /**
  * Tests the page display plugin.
@@ -57,7 +54,7 @@ protected function setUp() {
     parent::setUp();
 
     // Setup the needed tables in order to make the drupal router working.
-    $this->installSchema('system', array('router', 'menu_router', 'url_alias'));
+    $this->installSchema('system', array('menu_router', 'url_alias'));
     $this->installSchema('menu_link', 'menu_links');
   }
 
@@ -65,9 +62,6 @@ protected function setUp() {
    * Checks the behavior of the page for access denied/not found behaviors.
    */
   public function testPageResponses() {
-    // @todo Importing a route should fire a container rebuild.
-    $this->container->get('router.builder')->rebuild();
-
     $subrequest = Request::create('/test_page_display_403', 'GET');
     $response = $this->container->get('http_kernel')->handle($subrequest, HttpKernelInterface::SUB_REQUEST);
     $this->assertEqual($response->getStatusCode(), 403);
@@ -87,6 +81,9 @@ public function testPageResponses() {
     $view = views_get_view('test_page_display');
     // Disable the view, rebuild menu, and request the page again.
     $view->storage->disable()->save();
+    // Router rebuild would occur in a kernel terminate event so we need to
+    // simulate that here.
+    \Drupal::service('router.builder')->rebuildIfNeeded();
 
     $response = $this->container->get('http_kernel')->handle($subrequest, HttpKernelInterface::SUB_REQUEST);
     $this->assertEqual($response->getStatusCode(), 404);
diff --git a/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayTest.php b/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayTest.php
index ae966090e148..6b28d360f3f1 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayTest.php
@@ -250,6 +250,7 @@ public function testInvalidDisplayPlugins() {
 
     // Rebuild the router, and ensure that the path is not accessible anymore.
     views_invalidate_cache();
+    \Drupal::service('router.builder')->rebuildIfNeeded();
 
     $this->drupalGet('test_display_invalid');
     $this->assertResponse(404);
diff --git a/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayUnitTest.php b/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayUnitTest.php
index 6315015ca3bf..aa7306c9ec6f 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayUnitTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Plugin/DisplayUnitTest.php
@@ -25,7 +25,7 @@ class DisplayUnitTest extends ViewUnitTestBase {
    *
    * @var array
    */
-  public static $modules = array('block', 'node');
+  public static $modules = array('block', 'node', 'field', 'user');
 
   /**
    * Views plugin types to test.
diff --git a/core/modules/views/lib/Drupal/views/Tests/Plugin/ExposedFormTest.php b/core/modules/views/lib/Drupal/views/Tests/Plugin/ExposedFormTest.php
index e3e3db08dbbc..2ad29c96c88b 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Plugin/ExposedFormTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Plugin/ExposedFormTest.php
@@ -67,8 +67,6 @@ public function testSubmitButton() {
     $view->display_handler->setOption('exposed_form', $exposed_form);
     $view->save();
 
-    views_invalidate_cache();
-
     // Make sure the submit button label changed.
     $this->drupalGet('test_exposed_form_buttons');
     $this->helperButtonHasLabel('edit-submit-test-exposed-form-buttons', $expected_label);
@@ -82,8 +80,6 @@ public function testSubmitButton() {
     $view->display_handler->setOption('exposed_form', $exposed_form);
     $view->save();
 
-    views_invalidate_cache();
-
     // Make sure the submit button label shows 'Apply'.
     $this->drupalGet('test_exposed_form_buttons');
     $this->helperButtonHasLabel('edit-submit-test-exposed-form-buttons', t('Apply'));
@@ -120,8 +116,6 @@ public function testResetButton() {
     $view->display_handler->setOption('exposed_form', $exposed_form);
     $view->save();
 
-    views_invalidate_cache();
-
     // Look whether the reset button label changed.
     $this->drupalGet('test_exposed_form_buttons', array('query' => array('type' => 'article')));
     $this->assertResponse(200);
diff --git a/core/modules/views/lib/Drupal/views/Tests/Plugin/MiniPagerTest.php b/core/modules/views/lib/Drupal/views/Tests/Plugin/MiniPagerTest.php
index b18e0d302ecb..10955d443a13 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Plugin/MiniPagerTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Plugin/MiniPagerTest.php
@@ -49,7 +49,6 @@ protected function setUp() {
    * Tests the rendering of mini pagers.
    */
   public function testMiniPagerRender() {
-    menu_router_rebuild();
     $this->drupalGet('test_mini_pager');
     $this->assertText('›› test', 'The next link appears on the first page.');
     $this->assertText('Page 1', 'The current page info shows the first page.');
diff --git a/core/modules/views/lib/Drupal/views/Tests/Plugin/RowEntityTest.php b/core/modules/views/lib/Drupal/views/Tests/Plugin/RowEntityTest.php
index d71c28315593..ede97bd0799a 100644
--- a/core/modules/views/lib/Drupal/views/Tests/Plugin/RowEntityTest.php
+++ b/core/modules/views/lib/Drupal/views/Tests/Plugin/RowEntityTest.php
@@ -51,7 +51,7 @@ public static function getInfo() {
   protected function setUp() {
     parent::setUp();
 
-    $this->installSchema('system', array('menu_router', 'router'));
+    $this->installSchema('system', array('menu_router'));
     $this->installSchema('taxonomy', array('taxonomy_term_data', 'taxonomy_term_hierarchy'));
     $this->installConfig(array('taxonomy'));
   }
diff --git a/core/modules/views/lib/Drupal/views/Tests/ViewUnitTestBase.php b/core/modules/views/lib/Drupal/views/Tests/ViewUnitTestBase.php
index e5ec825710b6..620e4fe14fd1 100644
--- a/core/modules/views/lib/Drupal/views/Tests/ViewUnitTestBase.php
+++ b/core/modules/views/lib/Drupal/views/Tests/ViewUnitTestBase.php
@@ -54,6 +54,9 @@ protected function setUpFixtures() {
       $this->installSchema('views_test_data', $table);
     }
 
+    // The router table is required for router rebuilds.
+    $this->installSchema('system', array('router'));
+
     // Load the test dataset.
     $data_set = $this->dataSet();
     $query = db_insert('views_test_data')
diff --git a/core/modules/views/views.module b/core/modules/views/views.module
index fe427b062d27..1a20ac158958 100644
--- a/core/modules/views/views.module
+++ b/core/modules/views/views.module
@@ -551,20 +551,13 @@ function views_invalidate_cache() {
   Cache::deleteTags(array('content' => TRUE));
 
   // Set the menu as needed to be rebuilt.
-  \Drupal::state()->set('menu_rebuild_needed', TRUE);
+  \Drupal::service('router.builder')->setRebuildNeeded();
 
   $module_handler = \Drupal::moduleHandler();
 
   // Reset the RouteSubscriber from views.
   \Drupal::getContainer()->get('views.route_subscriber')->reset();
 
-  // Set the router to be rebuild.
-  // @todo Figure out why the cache rebuild is trigged but the route table
-  //   does not exist yet.
-  if (db_table_exists('router')) {
-    \Drupal::service('router.builder')->rebuild();
-  }
-
   // Invalidate the block cache to update views block derivatives.
   if ($module_handler->moduleExists('block')) {
     \Drupal::service('plugin.manager.block')->clearCachedDefinitions();
diff --git a/core/modules/views_ui/lib/Drupal/views_ui/Tests/ExposedFormUITest.php b/core/modules/views_ui/lib/Drupal/views_ui/Tests/ExposedFormUITest.php
index f5aec907e3ca..a17ffce12e05 100644
--- a/core/modules/views_ui/lib/Drupal/views_ui/Tests/ExposedFormUITest.php
+++ b/core/modules/views_ui/lib/Drupal/views_ui/Tests/ExposedFormUITest.php
@@ -43,7 +43,6 @@ protected function setUp() {
    * Tests the admin interface of exposed filter and sort items.
    */
   function testExposedAdminUi() {
-    menu_router_rebuild();
     $edit = array();
 
     $this->drupalGet('admin/structure/views/nojs/handler/test_exposed_admin_ui/default/filter/type');
diff --git a/core/modules/views_ui/lib/Drupal/views_ui/Tests/SettingsTest.php b/core/modules/views_ui/lib/Drupal/views_ui/Tests/SettingsTest.php
index 764bb0b8306e..86029a999183 100644
--- a/core/modules/views_ui/lib/Drupal/views_ui/Tests/SettingsTest.php
+++ b/core/modules/views_ui/lib/Drupal/views_ui/Tests/SettingsTest.php
@@ -87,7 +87,7 @@ function testEditUI() {
       'ui_show_display_embed' => FALSE,
     );
     $this->drupalPostForm('admin/structure/views/settings', $edit, t('Save configuration'));
-    views_invalidate_cache();
+
     $this->drupalPostForm('admin/structure/views/add', $view, t('Save and edit'));
     $this->assertNoFieldById('edit-displays-top-add-display-embed');
 
diff --git a/core/tests/Drupal/Tests/Core/Routing/NullRouteBuilder.php b/core/tests/Drupal/Tests/Core/Routing/NullRouteBuilder.php
new file mode 100644
index 000000000000..2631c480a0d0
--- /dev/null
+++ b/core/tests/Drupal/Tests/Core/Routing/NullRouteBuilder.php
@@ -0,0 +1,18 @@
+<?php
+
+namespace Drupal\Tests\Core\Routing;
+
+use Drupal\Core\Routing\RouteBuilderInterface;
+
+class NullRouteBuilder implements RouteBuilderInterface {
+
+  public function rebuild() {
+  }
+
+  public function rebuildIfNeeded() {
+  }
+
+  public function setRebuildNeeded() {
+  }
+
+}
diff --git a/core/tests/Drupal/Tests/Core/Routing/RouteBuilderTest.php b/core/tests/Drupal/Tests/Core/Routing/RouteBuilderTest.php
index 98cbfdad244d..9e05a5c080a0 100644
--- a/core/tests/Drupal/Tests/Core/Routing/RouteBuilderTest.php
+++ b/core/tests/Drupal/Tests/Core/Routing/RouteBuilderTest.php
@@ -75,6 +75,13 @@ class RouteBuilderTest extends UnitTestCase {
    */
   protected $controllerResolver;
 
+  /**
+   * The key value store.
+   *
+   * @var \Drupal\Core\KeyValueStore\StateInterface|\PHPUnit_Framework_MockObject_MockObject
+   */
+  protected $state;
+
   public static function getInfo() {
     return array(
       'name' => 'Route Builder',
@@ -92,8 +99,9 @@ protected function setUp() {
     $this->yamlDiscovery = $this->getMockBuilder('\Drupal\Component\Discovery\YamlDiscovery')
       ->disableOriginalConstructor()
       ->getMock();
+    $this->state = $this->getMock('\Drupal\Core\KeyValueStore\StateInterface');
 
-    $this->routeBuilder = new TestRouteBuilder($this->dumper, $this->lock, $this->dispatcher, $this->moduleHandler, $this->controllerResolver);
+    $this->routeBuilder = new TestRouteBuilder($this->dumper, $this->lock, $this->dispatcher, $this->moduleHandler, $this->controllerResolver, $this->state);
     $this->routeBuilder->setYamlDiscovery($this->yamlDiscovery);
   }
 
@@ -110,6 +118,10 @@ public function testRebuildLockingUnlocking() {
       ->method('release')
       ->with('router_rebuild');
 
+    $this->state->expects($this->once())
+       ->method('delete')
+       ->with('router_rebuild_needed');
+
     $this->yamlDiscovery->expects($this->any())
       ->method('findAll')
       ->will($this->returnValue(array()));
@@ -254,6 +266,44 @@ public function testRebuildWithProviderBasedRoutes() {
     $this->assertTrue($this->routeBuilder->rebuild());
   }
 
+  /**
+   * Tests \Drupal\Core\Routing\RouteBuilder::rebuildIfNeeded() method.
+   */
+  public function testRebuildIfNecessary() {
+    $this->lock->expects($this->once())
+               ->method('acquire')
+               ->with('router_rebuild')
+               ->will($this->returnValue(TRUE));
+
+    $this->lock->expects($this->once())
+               ->method('release')
+               ->with('router_rebuild');
+
+    $this->state->expects($this->once())
+                ->method('set')
+                ->with('router_rebuild_needed');
+
+    $this->state->expects($this->once())
+                ->method('delete')
+                ->with('router_rebuild_needed');
+
+    $this->state->expects($this->exactly(2))
+                ->method('get')
+                ->with('router_rebuild_needed')
+                ->will($this->onConsecutiveCalls(TRUE, FALSE));
+
+    $this->yamlDiscovery->expects($this->any())
+                        ->method('findAll')
+                        ->will($this->returnValue(array()));
+
+    $this->routeBuilder->setRebuildNeeded();
+
+    // This will trigger a successful rebuild.
+    $this->assertTrue($this->routeBuilder->rebuildIfNeeded());
+
+    // This will not trigger a rebuild.
+    $this->assertFalse($this->routeBuilder->rebuildIfNeeded());
+  }
 }
 
 /**
-- 
GitLab