diff --git a/core/core.services.yml b/core/core.services.yml
index a7a2b3bdbce9a606ca54015f55de37c7febf5a64..8ce9b11e3b858e73cbcd618ffdbbd1d446eef116 100644
--- a/core/core.services.yml
+++ b/core/core.services.yml
@@ -6,26 +6,86 @@ parameters:
   factory.keyvalue.expirable:
     default: keyvalue.expirable.database
 services:
-  cache_factory:
-    class: Drupal\Core\Cache\CacheFactory
-    arguments: ['@settings', '%cache_default_bin_backends%']
-    calls:
-      - [setContainer, ['@service_container']]
-  cache_contexts:
-    class: Drupal\Core\Cache\CacheContexts
-    arguments: ['@service_container', '%cache_contexts%' ]
+  # Simple cache contexts, directly derived from the request context.
+  cache_context.ip:
+    class: Drupal\Core\Cache\IpCacheContext
+    arguments: ['@request_stack']
+    tags:
+      - { name: cache.context }
+  cache_context.headers:
+    class: Drupal\Core\Cache\HeadersCacheContext
+    arguments: ['@request_stack']
+    tags:
+      - { name: cache.context }
+  cache_context.cookies:
+    class: Drupal\Core\Cache\CookiesCacheContext
+    arguments: ['@request_stack']
+    tags:
+      - { name: cache.context }
+  cache_context.request_format:
+    class: Drupal\Core\Cache\RequestFormatCacheContext
+    arguments: ['@request_stack']
+    tags:
+      - { name: cache.context }
   cache_context.url:
     class: Drupal\Core\Cache\UrlCacheContext
     arguments: ['@request_stack']
     tags:
-      - { name: cache.context}
-  cache_context.pager:
-    class: Drupal\Core\Cache\PagerCacheContext
+      - { name: cache.context }
+  cache_context.url.host:
+    class: Drupal\Core\Cache\HostCacheContext
     arguments: ['@request_stack']
+    tags:
+      - { name: cache.context }
+  cache_context.url.query_args:
+    class: Drupal\Core\Cache\QueryArgsCacheContext
+    arguments: ['@request_stack']
+    tags:
+      - { name: cache.context }
+  cache_context.url.query_args.pagers:
+    class: Drupal\Core\Cache\PagersCacheContext
+    arguments: ['@request_stack']
+    tags:
+      - { name: cache.context }
+
+  # Complex cache contexts, that depend on the routing system.
+  cache_context.route:
+    class: Drupal\Core\Cache\RouteCacheContext
+    arguments: ['@current_route_match']
+    tags:
+      - { name: cache.context }
+  cache_context.route.name:
+    class: Drupal\Core\Cache\RouteNameCacheContext
+    arguments: ['@current_route_match']
+    tags:
+      - { name: cache.context }
+  cache_context.route.menu_active_trails:
+    class: Drupal\Core\Cache\MenuActiveTrailsCacheContext
+    calls:
+      - [setContainer, ['@service_container']]
+    tags:
+      - { name: cache.context }
+
+  # Complex cache contexts, that may be calculated from a combination of
+  # multiple aspects of the request context plus additional logic. Hence they
+  # are their own roots.
+  cache_context.user:
+    class: Drupal\Core\Cache\UserCacheContext
+    arguments: ['@current_user']
+    tags:
+      - { name: cache.context}
+  cache_context.user.roles:
+    class: Drupal\Core\Cache\UserRolesCacheContext
+    arguments: ['@current_user']
     tags:
       - { name: cache.context}
-  cache_context.language:
-    class: Drupal\Core\Cache\LanguageCacheContext
+  cache_context.user.is_super_user:
+    class: Drupal\Core\Cache\IsSuperUserCacheContext
+    arguments: ['@current_user']
+    tags:
+      - { name: cache.context}
+  cache_context.languages:
+    class: Drupal\Core\Cache\LanguagesCacheContext
     arguments: ['@language_manager']
     tags:
       - { name: cache.context}
@@ -38,12 +98,15 @@ services:
     class: Drupal\Core\Cache\TimeZoneCacheContext
     tags:
       - { name: cache.context}
-  cache_context.menu.active_trail:
-    class: Drupal\Core\Cache\MenuActiveTrailCacheContext
+
+  cache_factory:
+    class: Drupal\Core\Cache\CacheFactory
+    arguments: ['@settings', '%cache_default_bin_backends%']
     calls:
       - [setContainer, ['@service_container']]
-    tags:
-      - { name: cache.context}
+  cache_contexts:
+    class: Drupal\Core\Cache\CacheContexts
+    arguments: ['@service_container', '%cache_contexts%' ]
   cache_tags.invalidator:
     parent: container.trait
     class: Drupal\Core\Cache\CacheTagsInvalidator
@@ -779,7 +842,7 @@ services:
       - { name: event_subscriber }
   main_content_renderer.html:
     class: Drupal\Core\Render\MainContent\HtmlRenderer
-    arguments: ['@title_resolver', '@plugin.manager.display_variant', '@event_dispatcher', '@module_handler', '@renderer']
+    arguments: ['@title_resolver', '@plugin.manager.display_variant', '@event_dispatcher', '@module_handler', '@renderer', '@cache_contexts']
     tags:
       - { name: render.main_content_renderer, format: html }
   main_content_renderer.ajax:
diff --git a/core/lib/Drupal/Core/Block/BlockBase.php b/core/lib/Drupal/Core/Block/BlockBase.php
index b8a7999d2fd34d910ab60c177de81c8b859b41d1..9872481caa87b147ca82ea79e66bfcf6b14790ec 100644
--- a/core/lib/Drupal/Core/Block/BlockBase.php
+++ b/core/lib/Drupal/Core/Block/BlockBase.php
@@ -207,7 +207,7 @@ public function buildConfigurationForm(array $form, FormStateInterface $form_sta
     $contexts = \Drupal::service("cache_contexts")->getLabels();
     // Blocks are always rendered in the "per language" and "per theme" cache
     // contexts. No need to show those options to the end user.
-    unset($contexts['language']);
+    unset($contexts['languages']);
     unset($contexts['theme']);
     $form['cache']['contexts'] = array(
       '#type' => 'checkboxes',
diff --git a/core/lib/Drupal/Core/Cache/CacheContexts.php b/core/lib/Drupal/Core/Cache/CacheContexts.php
index 33f86b1d91879a86456950931e4555a5a2b3b4ba..781f24a8a2257e25956756327a33580a70d32857 100644
--- a/core/lib/Drupal/Core/Cache/CacheContexts.php
+++ b/core/lib/Drupal/Core/Cache/CacheContexts.php
@@ -105,6 +105,7 @@ public function getLabels($include_calculated_cache_contexts = FALSE) {
    * @throws \InvalidArgumentException
    */
   public function convertTokensToKeys(array $context_tokens) {
+    $context_tokens = $this->optimizeTokens($context_tokens);
     sort($context_tokens);
     $keys = [];
     foreach (static::parseTokens($context_tokens) as $context) {
@@ -117,6 +118,69 @@ public function convertTokensToKeys(array $context_tokens) {
     return $keys;
   }
 
+  /**
+   * Optimizes cache context tokens (the minimal representative subset).
+   *
+   * A minimal representative subset means that any cache context token in the
+   * given set of cache context tokens that is a property of another cache
+   * context cache context token in the set, is removed.
+   *
+   * Hence a minimal representative subset is the most compact representation
+   * possible of a set of cache context tokens, that still captures the entire
+   * universe of variations.
+   *
+   * E.g. when caching per user ('user'), also caching per role ('user.roles')
+   * is meaningless because "per role" is implied by "per user".
+   *
+   * Examples — remember that the period indicates hierarchy and the colon can
+   * be used to get a specific value of a calculated cache context:
+   * - ['a', 'a.b'] -> ['a']
+   * - ['a', 'a.b.c'] -> ['a']
+   * - ['a.b', 'a.b.c'] -> ['a.b']
+   * - ['a', 'a.b', 'a.b.c'] -> ['a']
+   * - ['x', 'x:foo'] -> ['x']
+   * - ['a', 'a.b.c:bar'] -> ['a']
+   *
+   * @param string[] $context_tokens
+   *   A set of cache context tokens.
+   *
+   * @return string[]
+   *   A representative subset of the given set of cache context tokens..
+   */
+  public function optimizeTokens(array $context_tokens) {
+    $optimized_content_tokens = [];
+    foreach ($context_tokens as $context_token) {
+      // Context tokens without:
+      // - a period means they don't have a parent
+      // - a colon means they're not a specific value of a cache context
+      // hence no optimizations are possible.
+      if (strpos($context_token, '.') === FALSE && strpos($context_token, ':') === FALSE) {
+        $optimized_content_tokens[] = $context_token;
+      }
+      // The context token has a period or a colon. Iterate over all ancestor
+      // cache contexts. If one exists, omit the context token.
+      else {
+        $ancestor_found = FALSE;
+        // Treat a colon like a period, that allows us to consider 'a' the
+        // ancestor of 'a:foo', without any additional code for the colon.
+        $ancestor = str_replace(':', '.', $context_token);
+        do {
+          $ancestor = substr($ancestor, 0, strrpos($ancestor, '.'));
+          if (in_array($ancestor, $context_tokens)) {
+            // An ancestor cache context is in $context_tokens, hence this cache
+            // context is implied.
+            $ancestor_found = TRUE;
+          }
+
+        } while(!$ancestor_found && strpos($ancestor, '.') !== FALSE);
+        if (!$ancestor_found) {
+          $optimized_content_tokens[] = $context_token;
+        }
+      }
+    }
+    return $optimized_content_tokens;
+  }
+
   /**
    * Retrieves a cache context service from the container.
    *
diff --git a/core/lib/Drupal/Core/Cache/CacheContextsPass.php b/core/lib/Drupal/Core/Cache/CacheContextsPass.php
index b1cb74e6886fa0c8218f0e81307c248735063e28..576b2b568fca8592140319627b57218950a5e150 100644
--- a/core/lib/Drupal/Core/Cache/CacheContextsPass.php
+++ b/core/lib/Drupal/Core/Cache/CacheContextsPass.php
@@ -28,6 +28,20 @@ public function process(ContainerBuilder $container) {
       }
       $cache_contexts[] = substr($id, 14);
     }
+
+    // Validate.
+    sort($cache_contexts);
+    foreach ($cache_contexts as $id) {
+      // Validate the hierarchy of non-root-level cache contexts.
+      if (strpos($id, '.') !== FALSE) {
+        $parent = substr($id, 0, strrpos($id, '.'));
+        if (!in_array($parent, $cache_contexts)) {
+          throw new \InvalidArgumentException(sprintf('The service "%s" has an invalid service ID: the period indicates the hierarchy of cache contexts, therefore "%s" is considered the parent cache context, but no cache context service with that name was found.', $id, $parent));
+        }
+      }
+    }
+
+
     $container->setParameter('cache_contexts', $cache_contexts);
   }
 
diff --git a/core/lib/Drupal/Core/Cache/CalculatedCacheContextInterface.php b/core/lib/Drupal/Core/Cache/CalculatedCacheContextInterface.php
index 795a512e72f993a03931573389ed49b0cd323065..38114c363cf5716692eef149244c5b5094ad6b5b 100644
--- a/core/lib/Drupal/Core/Cache/CalculatedCacheContextInterface.php
+++ b/core/lib/Drupal/Core/Cache/CalculatedCacheContextInterface.php
@@ -28,12 +28,13 @@ public static function getLabel();
    * A cache context service's name is used as a token (placeholder) cache key,
    * and is then replaced with the string returned by this method.
    *
-   * @param string $parameter
-   *   The parameter.
+   * @param string|null $parameter
+   *   The parameter, or NULL to indicate all possible parameter values.
    *
    * @return string
-   *   The string representation of the cache context.
+   *   The string representation of the cache context. When $parameter is NULL,
+   *   a value representing all possible parameters must be generated.
    */
-  public function getContext($parameter);
+  public function getContext($parameter = NULL);
 
 }
diff --git a/core/lib/Drupal/Core/Cache/CookiesCacheContext.php b/core/lib/Drupal/Core/Cache/CookiesCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..6760b1b5c5a69e5fdc044705f650677b0e4f5456
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/CookiesCacheContext.php
@@ -0,0 +1,34 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\CookiesCacheContext.
+ */
+
+namespace Drupal\Core\Cache;
+
+/**
+ * Defines the CookiesCacheContext service, for "per cookie" caching.
+ */
+class CookiesCacheContext extends RequestStackCacheContextBase implements CalculatedCacheContextInterface {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t('HTTP cookies');
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getContext($cookie = NULL) {
+    if ($cookie === NULL) {
+      return $this->requestStack->getCurrentRequest()->cookies->all();
+    }
+    else {
+      return $this->requestStack->getCurrentRequest()->cookies->get($cookie);
+    }
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Cache/HeadersCacheContext.php b/core/lib/Drupal/Core/Cache/HeadersCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..e2b8932dab244e56c103fceeae7f1ffc4446f42c
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/HeadersCacheContext.php
@@ -0,0 +1,34 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\HeadersCacheContext.
+ */
+
+namespace Drupal\Core\Cache;
+
+/**
+ * Defines the HeadersCacheContext service, for "per header" caching.
+ */
+class HeadersCacheContext extends RequestStackCacheContextBase implements CalculatedCacheContextInterface {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t('HTTP headers');
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getContext($header = NULL) {
+    if ($header === NULL) {
+      return $this->requestStack->getCurrentRequest()->headers->all();
+    }
+    else {
+      return $this->requestStack->getCurrentRequest()->headers->get($header);
+    }
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Cache/HostCacheContext.php b/core/lib/Drupal/Core/Cache/HostCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..8a3af00bee156853fe0799b60ad6c1d603fc4608
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/HostCacheContext.php
@@ -0,0 +1,33 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\HostCacheContext.
+ */
+
+namespace Drupal\Core\Cache;
+
+/**
+ * Defines the HostCacheContext service, for "per host" caching.
+ *
+ * A "host" is defined as the combination of URI scheme, domain name and port.
+ *
+ * @see Symfony\Component\HttpFoundation::getSchemeAndHttpHost()
+ */
+class HostCacheContext extends RequestStackCacheContextBase {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t('Host');
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getContext() {
+    return $this->requestStack->getCurrentRequest()->getSchemeAndHttpHost();
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Cache/IpCacheContext.php b/core/lib/Drupal/Core/Cache/IpCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..c0c2e08020fb08a30a73be9e500a9921d3146044
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/IpCacheContext.php
@@ -0,0 +1,29 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\IpCacheContext.
+ */
+
+namespace Drupal\Core\Cache;
+
+/**
+ * Defines the IpCacheContext service, for "per IP address" caching.
+ */
+class IpCacheContext extends RequestStackCacheContextBase {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t('IP address');
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getContext() {
+    return $this->requestStack->getCurrentRequest()->getClientIp();
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Cache/IsSuperUserCacheContext.php b/core/lib/Drupal/Core/Cache/IsSuperUserCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..65697155357f7b0725573b3fb08be24141205195
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/IsSuperUserCacheContext.php
@@ -0,0 +1,29 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\IsSuperUserCacheContext.
+ */
+
+namespace Drupal\Core\Cache;
+
+/**
+ * Defines the IsSuperUserCacheContext service, for "super user or not" caching.
+ */
+class IsSuperUserCacheContext extends UserCacheContext {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t('Is super user');
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getContext() {
+    return ((int) $this->user->id()) === 1 ? '1' : '0';
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Cache/LanguageCacheContext.php b/core/lib/Drupal/Core/Cache/LanguageCacheContext.php
deleted file mode 100644
index b9d19a72d9c94f725b44fd116892c02f0de5bec6..0000000000000000000000000000000000000000
--- a/core/lib/Drupal/Core/Cache/LanguageCacheContext.php
+++ /dev/null
@@ -1,57 +0,0 @@
-<?php
-
-/**
- * @file
- * Contains \Drupal\Core\Cache\LanguageCacheContext.
- */
-
-namespace Drupal\Core\Cache;
-
-use Drupal\Core\Language\LanguageManagerInterface;
-
-/**
- * Defines the LanguageCacheContext service, for "per language" caching.
- */
-class LanguageCacheContext implements CacheContextInterface {
-
-  /**
-   * The language manager.
-   *
-   * @param \Drupal\Core\Language\LanguageManagerInterface $language_manager
-   */
-  protected $languageManager;
-
-  /**
-   * Constructs a new LanguageCacheContext service.
-   *
-   * @param \Drupal\Core\Language\LanguageManagerInterface $language_manager
-   *   The language manager.
-   */
-  public function __construct(LanguageManagerInterface $language_manager) {
-    $this->languageManager = $language_manager;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public static function getLabel() {
-    return t('Language');
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getContext() {
-    $context_parts = array();
-    if ($this->languageManager->isMultilingual()) {
-      foreach ($this->languageManager->getLanguageTypes() as $type) {
-        $context_parts[] = $this->languageManager->getCurrentLanguage($type)->getId();
-      }
-    }
-    else {
-      $context_parts[] = $this->languageManager->getCurrentLanguage()->getId();
-    }
-    return implode(':', $context_parts);
-  }
-
-}
diff --git a/core/lib/Drupal/Core/Cache/LanguagesCacheContext.php b/core/lib/Drupal/Core/Cache/LanguagesCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..2ed15177672d09ef2942321c13ad69bdf8384bea
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/LanguagesCacheContext.php
@@ -0,0 +1,76 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\LanguagesCacheContext.
+ */
+
+namespace Drupal\Core\Cache;
+
+use Drupal\Core\Language\LanguageManagerInterface;
+
+/**
+ * Defines the LanguagesCacheContext service, for "per language" caching.
+ */
+class LanguagesCacheContext implements CalculatedCacheContextInterface  {
+
+  /**
+   * The language manager.
+   *
+   * @param \Drupal\Core\Language\LanguageManagerInterface $language_manager
+   */
+  protected $languageManager;
+
+  /**
+   * Constructs a new LanguagesCacheContext service.
+   *
+   * @param \Drupal\Core\Language\LanguageManagerInterface $language_manager
+   *   The language manager.
+   */
+  public function __construct(LanguageManagerInterface $language_manager) {
+    $this->languageManager = $language_manager;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t('Language');
+  }
+
+  /**
+   * {@inheritdoc}
+   *
+   * $type can be NULL, or one of the language types supported by the language
+   * manager, typically:
+   * - LanguageInterface::TYPE_INTERFACE
+   * - LanguageInterface::TYPE_CONTENT
+   * - LanguageInterface::TYPE_URL
+   *
+   * @see \Drupal\Core\Language\LanguageManagerInterface::getLanguageTypes()
+   *
+   * @throws \RuntimeException
+   *   In case an invalid language type is specified.
+   */
+  public function getContext($type = NULL) {
+    if ($type === NULL) {
+      $context_parts = array();
+      if ($this->languageManager->isMultilingual()) {
+        foreach ($this->languageManager->getLanguageTypes() as $type) {
+          $context_parts[] = $this->languageManager->getCurrentLanguage($type)->getId();
+        }
+      }
+      else {
+        $context_parts[] = $this->languageManager->getCurrentLanguage()->getId();
+      }
+      return implode(',', $context_parts);
+    }
+    else {
+      if (!in_array($type, $this->languageManager->getLanguageTypes())) {
+        throw new \RuntimeException(sprintf('The language type "%s" is invalid.', $type));
+      }
+      return $this->languageManager->getCurrentLanguage($type)->getId();
+    }
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Cache/MenuActiveTrailCacheContext.php b/core/lib/Drupal/Core/Cache/MenuActiveTrailsCacheContext.php
similarity index 68%
rename from core/lib/Drupal/Core/Cache/MenuActiveTrailCacheContext.php
rename to core/lib/Drupal/Core/Cache/MenuActiveTrailsCacheContext.php
index 7c41bee2a8e00922eb4bece18a28e9ce8bdc6daa..3b43178a31b8c0dbc37db1355e92b82212d244cd 100644
--- a/core/lib/Drupal/Core/Cache/MenuActiveTrailCacheContext.php
+++ b/core/lib/Drupal/Core/Cache/MenuActiveTrailsCacheContext.php
@@ -2,7 +2,7 @@
 
 /**
  * @file
- * Contains \Drupal\Core\Cache\MenuActiveTrailCacheContext.
+ * Contains \Drupal\Core\Cache\MenuActiveTrailsCacheContext.
  */
 
 namespace Drupal\Core\Cache;
@@ -10,12 +10,12 @@
 use Symfony\Component\DependencyInjection\ContainerAware;
 
 /**
- * Defines the MenuActiveTrailCacheContext service.
+ * Defines the MenuActiveTrailsCacheContext service.
  *
  * This class is container-aware to avoid initializing the 'menu.active_trail'
  * service (and its dependencies) when it is not necessary.
  */
-class MenuActiveTrailCacheContext extends ContainerAware implements CalculatedCacheContextInterface {
+class MenuActiveTrailsCacheContext extends ContainerAware implements CalculatedCacheContextInterface {
 
   /**
    * {@inheritdoc}
@@ -27,7 +27,7 @@ public static function getLabel() {
   /**
    * {@inheritdoc}
    */
-  public function getContext($menu_name) {
+  public function getContext($menu_name = NULL) {
     $active_trail = $this->container->get('menu.active_trail')
       ->getActiveTrailIds($menu_name);
     return 'menu_trail.' . $menu_name . '|' . implode('|', $active_trail);
diff --git a/core/lib/Drupal/Core/Cache/PagerCacheContext.php b/core/lib/Drupal/Core/Cache/PagerCacheContext.php
deleted file mode 100644
index a4c308c84b31ed860aceef7db25a820fe64be8ad..0000000000000000000000000000000000000000
--- a/core/lib/Drupal/Core/Cache/PagerCacheContext.php
+++ /dev/null
@@ -1,29 +0,0 @@
-<?php
-
-/**
- * @file
- * Contains \Drupal\Core\Cache\PagerCacheContext.
- */
-
-namespace Drupal\Core\Cache;
-
-/**
- * Defines a cache context for "per page in a pager" caching.
- */
-class PagerCacheContext implements CalculatedCacheContextInterface {
-
-  /**
-   * {@inheritdoc}
-   */
-  public static function getLabel() {
-    return t('Pager');
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getContext($pager_id) {
-    return 'pager.' . $pager_id . '.' . pager_find_page($pager_id);
-  }
-
-}
diff --git a/core/lib/Drupal/Core/Cache/PagersCacheContext.php b/core/lib/Drupal/Core/Cache/PagersCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..cd9bb98895e3509f585ea5d70b5cd77b6201caab
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/PagersCacheContext.php
@@ -0,0 +1,37 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\PagersCacheContext.
+ */
+
+namespace Drupal\Core\Cache;
+
+/**
+ * Defines a cache context for "per page in a pager" caching.
+ */
+class PagersCacheContext extends RequestStackCacheContextBase implements CalculatedCacheContextInterface {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t('Pager');
+  }
+
+  /**
+   * {@inheritdoc}
+   *
+   * @see pager_find_page()
+   */
+  public function getContext($pager_id = NULL) {
+    // The value of the 'page' query argument contains the information that
+    // controls *all* pagers.
+    if ($pager_id === NULL) {
+      return 'pager' . $this->requestStack->getCurrentRequest()->query->get('page', '');
+    }
+
+    return 'pager.' . $pager_id . '.' . pager_find_page($pager_id);
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Cache/QueryArgsCacheContext.php b/core/lib/Drupal/Core/Cache/QueryArgsCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..4c34721d74d7aa51b8a206cfe165b2d601f1d7f7
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/QueryArgsCacheContext.php
@@ -0,0 +1,38 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\QueryArgsCacheContext.
+ */
+
+namespace Drupal\Core\Cache;
+
+/**
+ * Defines the QueryArgsCacheContext service, for "per query args" caching.
+ *
+ * A "host" is defined as the combination of URI scheme, domain name and port.
+ *
+ * @see Symfony\Component\HttpFoundation::getSchemeAndHttpHost()
+ */
+class QueryArgsCacheContext extends RequestStackCacheContextBase implements CalculatedCacheContextInterface {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t('Query arguments');
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getContext($query_arg = NULL) {
+    if ($query_arg === NULL) {
+      return $this->requestStack->getCurrentRequest()->getQueryString();
+    }
+    else {
+      return $this->requestStack->getCurrentRequest()->query->get($query_arg);
+    }
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Cache/RequestFormatCacheContext.php b/core/lib/Drupal/Core/Cache/RequestFormatCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..397384fbcf7c033cbc2cdf53a4aa097f9f9fce9a
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/RequestFormatCacheContext.php
@@ -0,0 +1,29 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\RequestFormatCacheContext.
+ */
+
+namespace Drupal\Core\Cache;
+
+/**
+ * Defines the RequestFormatCacheContext service, for "per format" caching.
+ */
+class RequestFormatCacheContext extends RequestStackCacheContextBase {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t('Request format');
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getContext() {
+    return $this->requestStack->getCurrentRequest()->getRequestFormat();
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Cache/RequestStackCacheContextBase.php b/core/lib/Drupal/Core/Cache/RequestStackCacheContextBase.php
new file mode 100644
index 0000000000000000000000000000000000000000..34e3560c2ba596a29e33c7789811ed907403689c
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/RequestStackCacheContextBase.php
@@ -0,0 +1,34 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\RequestStackCacheContextBase.
+ */
+
+namespace Drupal\Core\Cache;
+
+use Symfony\Component\HttpFoundation\RequestStack;
+
+/**
+ * Defines a base class for cache contexts depending only on the request stack.
+ */
+abstract class RequestStackCacheContextBase implements CacheContextInterface {
+
+  /**
+   * The request stack.
+   *
+   * @var \Symfony\Component\HttpFoundation\RequestStack
+   */
+  protected $requestStack;
+
+  /**
+   * Constructs a new RequestStackCacheContextBase class.
+   *
+   * @param \Symfony\Component\HttpFoundation\RequestStack $request_stack
+   *   The request stack.
+   */
+  public function __construct(RequestStack $request_stack) {
+    $this->requestStack = $request_stack;
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Cache/RouteCacheContext.php b/core/lib/Drupal/Core/Cache/RouteCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..6e9e39f4e9ed507b1d272df43b77cba34952f3bf
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/RouteCacheContext.php
@@ -0,0 +1,48 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\RouteCacheContext.
+ */
+
+namespace Drupal\Core\Cache;
+
+use Drupal\Core\Routing\RouteMatchInterface;
+
+/**
+ * Defines the RouteCacheContext service, for "per route" caching.
+ */
+class RouteCacheContext {
+
+  /**
+   * The route match.
+   *
+   * @var \Drupal\Core\Routing\RouteMatchInterface
+   */
+  protected $routeMatch;
+
+  /**
+   * Constructs a new RouteCacheContext class.
+   *
+   * @param \Drupal\Core\Routing\RouteMatchInterface $route_match
+   *   The route match.
+   */
+  public function __construct(RouteMatchInterface $route_match) {
+    $this->routeMatch = $route_match;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t('Route');
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getContext() {
+    return $this->routeMatch->getRouteName() . hash('sha256', serialize($this->routeMatch->getRawParameters()->all()));
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Cache/RouteNameCacheContext.php b/core/lib/Drupal/Core/Cache/RouteNameCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..f0bd93788108f8ff908705fcaffbdeedc09922dc
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/RouteNameCacheContext.php
@@ -0,0 +1,29 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\RouteNameCacheContext.
+ */
+
+namespace Drupal\Core\Cache;
+
+/**
+ * Defines the RouteCacheContext service, for "per route name" caching.
+ */
+class RouteNameCacheContext extends RouteCacheContext {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t('Route name');
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getContext() {
+    return $this->routeMatch->getRouteName();
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Cache/UrlCacheContext.php b/core/lib/Drupal/Core/Cache/UrlCacheContext.php
index 308d1827970330fd058126a7a93f832ce9160ed1..138109317d3890d95073ec030656d356b979e280 100644
--- a/core/lib/Drupal/Core/Cache/UrlCacheContext.php
+++ b/core/lib/Drupal/Core/Cache/UrlCacheContext.php
@@ -7,29 +7,10 @@
 
 namespace Drupal\Core\Cache;
 
-use Symfony\Component\HttpFoundation\RequestStack;
-
 /**
  * Defines the UrlCacheContext service, for "per page" caching.
  */
-class UrlCacheContext implements CacheContextInterface {
-
-  /**
-   * The request stack.
-   *
-   * @var \Symfony\Component\HttpFoundation\RequestStack
-   */
-  protected $requestStack;
-
-  /**
-   * Constructs a new UrlCacheContext service.
-   *
-   * @param \Symfony\Component\HttpFoundation\RequestStack $request_stack
-   *   The request stack.
-   */
-  public function __construct(RequestStack $request_stack) {
-    $this->requestStack = $request_stack;
-  }
+class UrlCacheContext extends RequestStackCacheContextBase {
 
   /**
    * {@inheritdoc}
diff --git a/core/modules/user/src/Cache/UserCacheContext.php b/core/lib/Drupal/Core/Cache/UserCacheContext.php
similarity index 83%
rename from core/modules/user/src/Cache/UserCacheContext.php
rename to core/lib/Drupal/Core/Cache/UserCacheContext.php
index d9d880ed700f2c593ad557a5ad05634abede6b9a..d5e282553979221d5b0fd6951950f69df64ce80b 100644
--- a/core/modules/user/src/Cache/UserCacheContext.php
+++ b/core/lib/Drupal/Core/Cache/UserCacheContext.php
@@ -2,12 +2,11 @@
 
 /**
  * @file
- * Contains \Drupal\user\Cache\UserCacheContext.
+ * Contains \Drupal\Core\Cache\UserCacheContext.
  */
 
-namespace Drupal\user\Cache;
+namespace Drupal\Core\Cache;
 
-use Drupal\Core\Cache\CacheContextInterface;
 use Drupal\Core\Session\AccountInterface;
 
 /**
diff --git a/core/lib/Drupal/Core/Cache/UserRolesCacheContext.php b/core/lib/Drupal/Core/Cache/UserRolesCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..cf7b4507b0f21521a1f590738cd9b60fc1ae9143
--- /dev/null
+++ b/core/lib/Drupal/Core/Cache/UserRolesCacheContext.php
@@ -0,0 +1,34 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\UserRolesCacheContext.
+ */
+
+namespace Drupal\Core\Cache;
+
+/**
+ * Defines the UserRolesCacheContext service, for "per role" caching.
+ */
+class UserRolesCacheContext extends UserCacheContext implements CalculatedCacheContextInterface{
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t("User's roles");
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getContext($role = NULL) {
+    if ($role === NULL) {
+      return 'r.' . implode(',', $this->user->getRoles());
+    }
+    else {
+      return 'r.' . $role . '.' . (in_array($role, $this->user->getRoles()) ? '0' : '1');
+    }
+  }
+
+}
diff --git a/core/lib/Drupal/Core/Field/Plugin/Field/FieldFormatter/LanguageFormatter.php b/core/lib/Drupal/Core/Field/Plugin/Field/FieldFormatter/LanguageFormatter.php
index d142872136bec765dd748ab85062384de3f3984c..fbf120619781f255651f455ccea67e0656ab15d6 100644
--- a/core/lib/Drupal/Core/Field/Plugin/Field/FieldFormatter/LanguageFormatter.php
+++ b/core/lib/Drupal/Core/Field/Plugin/Field/FieldFormatter/LanguageFormatter.php
@@ -117,7 +117,7 @@ public function settingsSummary() {
    * {@inheritdoc}
    */
   protected function viewValue(FieldItemInterface $item) {
-    // The language cache context is not necessary because the language is
+    // The 'languages' cache context is not necessary because the language is
     // either displayed in its configured form (loaded directly from config
     // storage by LanguageManager::getLanguages()) or in its native language
     // name. That only depends on formatter settings and no language condition.
diff --git a/core/lib/Drupal/Core/Menu/MenuActiveTrailInterface.php b/core/lib/Drupal/Core/Menu/MenuActiveTrailInterface.php
index 49f4e99d5b6337ecd0b417dafca5a52348bb0eef..be32e641df06bac60fac234f41480363b52a7c9f 100644
--- a/core/lib/Drupal/Core/Menu/MenuActiveTrailInterface.php
+++ b/core/lib/Drupal/Core/Menu/MenuActiveTrailInterface.php
@@ -18,8 +18,9 @@ interface MenuActiveTrailInterface {
   /**
    * Gets the active trail IDs of the specified menu tree.
    *
-   * @param string $menu_name
-   *   The menu name of the requested tree.
+   * @param string|NULL $menu_name
+   *   (optional) The menu name of the requested tree. If omitted, all menu
+   *   trees will be searched.
    *
    * @return array
    *   An array containing the active trail: a list of plugin IDs.
diff --git a/core/lib/Drupal/Core/Render/Element/Pager.php b/core/lib/Drupal/Core/Render/Element/Pager.php
index 006064a6d781759ea9f7cfd3a7756e1dd34ca9df..eeb0f9dd57655052b629f3de077f77003331ffdb 100644
--- a/core/lib/Drupal/Core/Render/Element/Pager.php
+++ b/core/lib/Drupal/Core/Render/Element/Pager.php
@@ -46,7 +46,7 @@ public function getInfo() {
    * @return array
    */
   public static function preRenderPager(array $pager) {
-    $pager['#cache']['contexts'][] = 'pager:' . $pager['#element'];
+    $pager['#cache']['contexts'][] = 'url.query_args.pagers:' . $pager['#element'];
     return $pager;
   }
 
diff --git a/core/lib/Drupal/Core/Render/MainContent/HtmlRenderer.php b/core/lib/Drupal/Core/Render/MainContent/HtmlRenderer.php
index 269b64e314562db602da0463bf7459ffac59fc56..249b47b0865e5189fd18990ac714eb89d45d8c31 100644
--- a/core/lib/Drupal/Core/Render/MainContent/HtmlRenderer.php
+++ b/core/lib/Drupal/Core/Render/MainContent/HtmlRenderer.php
@@ -10,6 +10,7 @@
 use Drupal\Component\Plugin\PluginManagerInterface;
 use Drupal\Component\Utility\NestedArray;
 use Drupal\Core\Cache\Cache;
+use Drupal\Core\Cache\CacheContexts;
 use Drupal\Core\Controller\TitleResolverInterface;
 use Drupal\Core\Display\PageVariantInterface;
 use Drupal\Core\Extension\ModuleHandlerInterface;
@@ -63,6 +64,13 @@ class HtmlRenderer implements MainContentRendererInterface {
    */
   protected $renderer;
 
+  /**
+   * The cache contexts service.
+   *
+   * @var \Drupal\Core\Cache\CacheContexts
+   */
+  protected $cacheContexts;
+
   /**
    * Constructs a new HtmlRenderer.
    *
@@ -76,13 +84,16 @@ class HtmlRenderer implements MainContentRendererInterface {
    *   The module handler.
    * @param \Drupal\Core\Render\RendererInterface $renderer
    *   The renderer service.
+   * @param \Drupal\Core\Cache\CacheContexts $cache_contexts
+   *   The cache contexts service.
    */
-  public function __construct(TitleResolverInterface $title_resolver, PluginManagerInterface $display_variant_manager, EventDispatcherInterface $event_dispatcher, ModuleHandlerInterface $module_handler, RendererInterface $renderer) {
+  public function __construct(TitleResolverInterface $title_resolver, PluginManagerInterface $display_variant_manager, EventDispatcherInterface $event_dispatcher, ModuleHandlerInterface $module_handler, RendererInterface $renderer, CacheContexts $cache_contexts) {
     $this->titleResolver = $title_resolver;
     $this->displayVariantManager = $display_variant_manager;
     $this->eventDispatcher = $event_dispatcher;
     $this->moduleHandler = $module_handler;
     $this->renderer = $renderer;
+    $this->cacheContexts = $cache_contexts;
   }
 
   /**
@@ -149,7 +160,7 @@ public function renderResponse(array $main_content, Request $request, RouteMatch
 
     $response = new Response($content, 200,[
       'X-Drupal-Cache-Tags' => implode(' ', $cache_tags),
-      'X-Drupal-Cache-Contexts' => implode(' ', $cache_contexts),
+      'X-Drupal-Cache-Contexts' => implode(' ', $this->cacheContexts->optimizeTokens($cache_contexts)),
       'X-Generator' => 'Drupal ' . $version . ' (https://www.drupal.org)'
     ]);
     // If an explicit non-infinite max-age is specified by a part of the page,
diff --git a/core/modules/block/src/BlockViewBuilder.php b/core/modules/block/src/BlockViewBuilder.php
index 8430393eab69d8b759b41860bbeaa04b941ba22a..d527d6d2b8affec0af63bdadb8c10e6a9c1c178e 100644
--- a/core/modules/block/src/BlockViewBuilder.php
+++ b/core/modules/block/src/BlockViewBuilder.php
@@ -87,7 +87,7 @@ public function viewMultiple(array $entities = array(), $view_mode = 'full', $la
           $entity->id(),
         );
         $default_cache_contexts = array(
-          'language',
+          'languages',
           'theme',
         );
         $max_age = $plugin->getCacheMaxAge();
diff --git a/core/modules/block/src/Tests/BlockInterfaceTest.php b/core/modules/block/src/Tests/BlockInterfaceTest.php
index 08980c5df91cf3611d205cf8fe08dbfd24aad530..37b9b3c5dab5843b4f4c008a60570de4948308fd 100644
--- a/core/modules/block/src/Tests/BlockInterfaceTest.php
+++ b/core/modules/block/src/Tests/BlockInterfaceTest.php
@@ -67,7 +67,7 @@ public function testBlockInterface() {
     $period[\Drupal\Core\Cache\Cache::PERMANENT] = t('Forever');
     $contexts = \Drupal::service("cache_contexts")->getLabels();
     unset($contexts['theme']);
-    unset($contexts['language']);
+    unset($contexts['languages']);
     $expected_form = array(
       'provider' => array(
         '#type' => 'value',
diff --git a/core/modules/block_content/src/Tests/BlockContentCacheTagsTest.php b/core/modules/block_content/src/Tests/BlockContentCacheTagsTest.php
index 16921baa5ad0dde54afbd251391047d096dd810b..d5a98f6b6e166508e47227c4cf681ad890e0c599 100644
--- a/core/modules/block_content/src/Tests/BlockContentCacheTagsTest.php
+++ b/core/modules/block_content/src/Tests/BlockContentCacheTagsTest.php
@@ -80,7 +80,7 @@ public function testBlock() {
     // Expected keys, contexts, and tags for the block.
     // @see \Drupal\block\BlockViewBuilder::viewMultiple()
     $expected_block_cache_keys = ['entity_view', 'block', $block->id()];
-    $expected_block_cache_contexts = ['language', 'theme'];
+    $expected_block_cache_contexts = ['languages', 'theme'];
     $expected_block_cache_tags = Cache::mergeTags(['block_view', 'rendered'], $block->getCacheTags(), $block->getPlugin()->getCacheTags());
 
     // Expected contexts and tags for the BlockContent entity.
diff --git a/core/modules/book/book.services.yml b/core/modules/book/book.services.yml
index 440d021aa3143c767b0967b30b55cf9b379b03b7..89b25fc02af1b0b10cdbfd74f8d0a3237d645ce6 100644
--- a/core/modules/book/book.services.yml
+++ b/core/modules/book/book.services.yml
@@ -23,7 +23,7 @@ services:
     arguments: ['@book.manager']
     tags:
       - { name: access_check, applies_to: _access_book_removable }
-  cache_context.book.navigation:
+  cache_context.route.book_navigation:
     class: Drupal\book\Cache\BookNavigationCacheContext
     arguments: ['@request_stack']
     tags:
diff --git a/core/modules/book/src/Plugin/Block/BookNavigationBlock.php b/core/modules/book/src/Plugin/Block/BookNavigationBlock.php
index 4ce4c3fa105dc6642712562ed6203d0552fe3b00..7b5279959ab6c9e14b1f53f0951ccbf87b52b29f 100644
--- a/core/modules/book/src/Plugin/Block/BookNavigationBlock.php
+++ b/core/modules/book/src/Plugin/Block/BookNavigationBlock.php
@@ -186,7 +186,7 @@ protected function getRequiredCacheContexts() {
     // context.
     return [
       'user.roles',
-      'book.navigation',
+      'route.book_navigation',
     ];
   }
 
diff --git a/core/modules/filter/src/Tests/FilterAPITest.php b/core/modules/filter/src/Tests/FilterAPITest.php
index 12efba397fff672b45f3ec132d16afac86cdd179..fe54ccdc8a9cb01bf09fb7af9570c818092ab141 100644
--- a/core/modules/filter/src/Tests/FilterAPITest.php
+++ b/core/modules/filter/src/Tests/FilterAPITest.php
@@ -8,6 +8,7 @@
 namespace Drupal\filter\Tests;
 
 use Drupal\Component\Utility\String;
+use Drupal\Core\Language\LanguageInterface;
 use Drupal\Core\Session\AnonymousUserSession;
 use Drupal\Core\TypedData\OptionsProviderInterface;
 use Drupal\Core\TypedData\DataDefinition;
@@ -260,7 +261,7 @@ function testProcessedTextElement() {
     $this->assertEqual($expected_cache_tags, $build['#cache']['tags'], 'Expected cache tags present.');
     $expected_cache_contexts = [
       // The cache context set by the filter_test_cache_contexts filter.
-      'language',
+      'languages:' . LanguageInterface::TYPE_CONTENT,
     ];
     $this->assertEqual($expected_cache_contexts, $build['#cache']['contexts'], 'Expected cache contexts present.');
     $expected_markup = '<p>Hello, world!</p><p>This is a dynamic llama.</p>';
diff --git a/core/modules/filter/tests/filter_test/src/Plugin/Filter/FilterTestCacheContexts.php b/core/modules/filter/tests/filter_test/src/Plugin/Filter/FilterTestCacheContexts.php
index 6a9895615685b527b117fe0e0b3bc74e7c2d9815..3cb6de2c89e7c8b170e7161d3ef6b4b009a717b8 100644
--- a/core/modules/filter/tests/filter_test/src/Plugin/Filter/FilterTestCacheContexts.php
+++ b/core/modules/filter/tests/filter_test/src/Plugin/Filter/FilterTestCacheContexts.php
@@ -7,6 +7,7 @@
 
 namespace Drupal\filter_test\Plugin\Filter;
 
+use Drupal\Core\Language\LanguageInterface;
 use Drupal\filter\FilterProcessResult;
 use Drupal\filter\Plugin\FilterBase;
 
@@ -28,7 +29,7 @@ class FilterTestCacheContexts extends FilterBase {
   public function process($text, $langcode) {
     $result = new FilterProcessResult($text);
     // The changes made by this filter are language-specific.
-    $result->addCacheContexts(['language']);
+    $result->addCacheContexts(['languages:' . LanguageInterface::TYPE_CONTENT]);
     return $result;
   }
 
diff --git a/core/modules/node/node.services.yml b/core/modules/node/node.services.yml
index b2585d9d009eabc4180143ec6e5412d6a4c00d1e..f3e54fb2fee38be41315356ffe3f2460e2c0384f 100644
--- a/core/modules/node/node.services.yml
+++ b/core/modules/node/node.services.yml
@@ -39,8 +39,8 @@ services:
     arguments: ['@current_route_match']
     tags:
       - { name: page_cache_response_policy }
-  cache_context.node_view_grants:
-    class: Drupal\node\Cache\NodeAccessViewGrantsCacheContext
+  cache_context.user.node_grants:
+    class: Drupal\node\Cache\NodeAccessGrantsCacheContext
     arguments: ['@current_user']
     tags:
       - { name: cache.context }
diff --git a/core/modules/node/src/Cache/NodeAccessGrantsCacheContext.php b/core/modules/node/src/Cache/NodeAccessGrantsCacheContext.php
new file mode 100644
index 0000000000000000000000000000000000000000..e213c40d62b647de623fb4f395bf185928cc880f
--- /dev/null
+++ b/core/modules/node/src/Cache/NodeAccessGrantsCacheContext.php
@@ -0,0 +1,80 @@
+<?php
+
+/**
+ * @file
+ * Contains \Drupal\Core\Cache\NodeAccessGrantsCacheContext.
+ */
+
+namespace Drupal\node\Cache;
+
+use Drupal\Core\Cache\CalculatedCacheContextInterface;
+use Drupal\Core\Cache\UserCacheContext;
+
+/**
+ * Defines the node access view cache context service.
+ *
+ * This allows for node access grants-sensitive caching when listing nodes.
+ *
+ * @see node_query_node_access_alter()
+ */
+class NodeAccessGrantsCacheContext extends UserCacheContext implements CalculatedCacheContextInterface {
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function getLabel() {
+    return t("Content access view grants");
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getContext($operation = NULL) {
+    // If the current user either can bypass node access then we don't need to
+    // determine the exact node grants for the current user.
+    if ($this->user->hasPermission('bypass node access')) {
+      return 'all';
+    }
+
+    // When no specific operation is specified, check the grants for all three
+    // possible operations.
+    if ($operation === NULL) {
+      $result = [];
+      foreach (['view', 'update', 'delete'] as $op) {
+        $result[] = $this->checkNodeGrants($op);
+      }
+      return implode('-', $result);
+    }
+    else {
+      return $this->checkNodeGrants($operation);
+    }
+  }
+
+  /**
+   * Checks the node grants for the given operation.
+   *
+   * @param string $operation
+   *   The operation to check the node grants for.
+   *
+   * @return string
+   *   The string representation of the cache context.
+   */
+  protected function checkNodeGrants($operation) {
+    // When checking the grants for the 'view' operation and the current user
+    // has a global view grant (i.e. a view grant for node ID 0) — note that
+    // this is automatically the case if no node access modules exist (no
+    // hook_node_grants() implementations) then we don't need to determine the
+    // exact node view grants for the current user.
+    if ($operation === 'view' && node_access_view_all_nodes($this->user)) {
+      return 'view.all';
+    }
+
+    $grants = node_access_grants($operation, $this->user);
+    $grants_context_parts = [];
+    foreach ($grants as $realm => $gids) {
+      $grants_context_parts[] = $realm . ':' . implode(',', $gids);
+    }
+    return $operation . '.' . implode(';', $grants_context_parts);
+  }
+
+}
diff --git a/core/modules/node/src/Cache/NodeAccessViewGrantsCacheContext.php b/core/modules/node/src/Cache/NodeAccessViewGrantsCacheContext.php
deleted file mode 100644
index 24323b9c42114149648246539ff530d5bca784d3..0000000000000000000000000000000000000000
--- a/core/modules/node/src/Cache/NodeAccessViewGrantsCacheContext.php
+++ /dev/null
@@ -1,69 +0,0 @@
-<?php
-
-/**
- * @file
- * Contains \Drupal\Core\Cache\NodeAccessViewGrantsCacheContext.
- */
-
-namespace Drupal\node\Cache;
-
-use Drupal\Core\Cache\CacheContextInterface;
-use Drupal\Core\Session\AccountInterface;
-
-/**
- * Defines the node access view grants cache context service.
- *
- * This allows for node access grants-sensitive caching when viewing nodes.
- *
- * @see node_query_node_access_alter()
- */
-class NodeAccessViewGrantsCacheContext implements CacheContextInterface {
-
-  /**
-   * The current user.
-   *
-   * @var \Drupal\Core\Session\AccountInterface
-   */
-  protected $user;
-
-  /**
-   * Constructs a new NodeAccessViewGrantsCacheContext service.
-   *
-   * @param \Drupal\Core\Session\AccountInterface $user
-   *   The current user.
-   */
-  public function __construct(AccountInterface $user) {
-    $this->user = $user;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public static function getLabel() {
-    return t("Content access view grants");
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getContext() {
-    // If the current user either:
-    // - can bypass node access
-    // - has a global view grant (such as a view grant for node ID 0) — note
-    //   that this is automatically the case if no node access modules exist (no
-    //   hook_node_grants() implementations)
-    // then we don't need to determine the exact node view grants for the
-    // current user.
-    if ($this->user->hasPermission('bypass node access') || node_access_view_all_nodes($this->user)) {
-      return 'all';
-    }
-
-    $grants = node_access_grants('view', $this->user);
-    $grants_context_parts = [];
-    foreach ($grants as $realm => $gids) {
-      $grants_context_parts[] = $realm . ':' . implode(',', $gids);
-    }
-    return implode(';', $grants_context_parts);
-  }
-
-}
diff --git a/core/modules/node/src/Entity/Node.php b/core/modules/node/src/Entity/Node.php
index eb99636dc4fe02fb002b267cf1f4b7398865b8a4..2621d82a2dd17268bf4aa533de823f1d20091b21 100644
--- a/core/modules/node/src/Entity/Node.php
+++ b/core/modules/node/src/Entity/Node.php
@@ -45,7 +45,7 @@
  *   revision_table = "node_revision",
  *   revision_data_table = "node_field_revision",
  *   translatable = TRUE,
- *   list_cache_contexts = { "node_view_grants" },
+ *   list_cache_contexts = { "user.node_grants:view" },
  *   entity_keys = {
  *     "id" = "nid",
  *     "revision" = "vid",
diff --git a/core/modules/node/src/NodeGrantDatabaseStorage.php b/core/modules/node/src/NodeGrantDatabaseStorage.php
index da6522dc0efa0a8686e23ed3f362b30a3d727f9f..5c661900468560404cb22c3bcec6149eec2aae16 100644
--- a/core/modules/node/src/NodeGrantDatabaseStorage.php
+++ b/core/modules/node/src/NodeGrantDatabaseStorage.php
@@ -106,12 +106,11 @@ public function access(NodeInterface $node, $operation, $langcode, AccountInterf
     // theoretically cacheable, because we don't have the necessary metadata to
     // know it for a fact.
     $set_cacheability = function (AccessResult $access_result) use ($operation) {
-      if ($operation === 'view') {
-        return $access_result->addCacheContexts(['node_view_grants']);
-      }
-      else {
-        return $access_result->setCacheable(FALSE);
+      $access_result->addCacheContexts(['user.node_grants:' . $operation]);
+      if ($operation !== 'view') {
+        $access_result->setCacheable(FALSE);
       }
+      return $access_result;
     };
 
     if ($query->execute()->fetchField()) {
diff --git a/core/modules/node/src/Plugin/views/argument_default/Node.php b/core/modules/node/src/Plugin/views/argument_default/Node.php
index 4a61f032a2b5410fe86b250a6bab6a8f164a0455..e6d8343233c508feb86651cc9b1ab0fbb7bee451 100644
--- a/core/modules/node/src/Plugin/views/argument_default/Node.php
+++ b/core/modules/node/src/Plugin/views/argument_default/Node.php
@@ -83,7 +83,7 @@ public function isCacheable() {
    * {@inheritdoc}
    */
   public function getCacheContexts() {
-    return ['cache.context.url'];
+    return ['url'];
   }
 
 }
diff --git a/core/modules/node/src/Tests/NodeAccessViewGrantsCacheContextTest.php b/core/modules/node/src/Tests/NodeAccessGrantsCacheContextTest.php
similarity index 76%
rename from core/modules/node/src/Tests/NodeAccessViewGrantsCacheContextTest.php
rename to core/modules/node/src/Tests/NodeAccessGrantsCacheContextTest.php
index 1cd8062dab41ca73eaaa49daccca786000a5f39c..d66e7769eb53d7d51fdd8685f8b04789b41b279c 100644
--- a/core/modules/node/src/Tests/NodeAccessViewGrantsCacheContextTest.php
+++ b/core/modules/node/src/Tests/NodeAccessGrantsCacheContextTest.php
@@ -2,18 +2,18 @@
 
 /**
  * @file
- * Contains \Drupal\node\Tests\NodeAccessViewGrantsCacheContextTest.
+ * Contains \Drupal\node\Tests\NodeAccessGrantsCacheContextTest.
  */
 
 namespace Drupal\node\Tests;
 
 /**
- * Tests the node access view grants cache context service.
+ * Tests the node access grants cache context service.
  *
  * @group node
  * @group Cache
  */
-class NodeAccessViewGrantsCacheContextTest extends NodeTestBase {
+class NodeAccessGrantsCacheContextTest extends NodeTestBase {
 
   /**
    * Modules to enable.
@@ -71,20 +71,20 @@ protected function assertCacheContext(array $expected) {
         $this->drupalLogin($this->userMapping[$uid]);
       }
       $this->pass('Asserting cache context for user ' . $uid . '.');
-      $this->assertIdentical($context, $this->container->get('cache_context.node_view_grants')->getContext());
+      $this->assertIdentical($context, $this->container->get('cache_context.user.node_grants')->getContext('view'));
     }
     $this->drupalLogout();
   }
 
   /**
-   * Tests NodeAccessViewGrantsCacheContext::getContext().
+   * Tests NodeAccessGrantsCacheContext::getContext().
    */
   public function testCacheContext() {
     $this->assertCacheContext([
-      0 => 'all:0;node_access_test_author:0;node_access_all:0',
+      0 => 'view.all:0;node_access_test_author:0;node_access_all:0',
       1 => 'all',
-      2 => 'all:0;node_access_test_author:2;node_access_test:8888,8889',
-      3 => 'all:0;node_access_test_author:3',
+      2 => 'view.all:0;node_access_test_author:2;node_access_test:8888,8889',
+      3 => 'view.all:0;node_access_test_author:3',
     ]);
 
     // Grant view to all nodes (because nid = 0) for users in the
@@ -103,40 +103,40 @@ public function testCacheContext() {
     \Drupal::state()->set('node_access_test.no_access_uid', 0);
     drupal_static_reset('node_access_view_all_nodes');
     $this->assertCacheContext([
-      0 => 'all',
+      0 => 'view.all',
       1 => 'all',
-      2 => 'all:0;node_access_test_author:2;node_access_test:8888,8889',
-      3 => 'all:0;node_access_test_author:3',
+      2 => 'view.all:0;node_access_test_author:2;node_access_test:8888,8889',
+      3 => 'view.all:0;node_access_test_author:3',
     ]);
 
     // Put user accessUser (uid 2) in the realm.
     \Drupal::state()->set('node_access_test.no_access_uid', $this->accessUser->id());
     drupal_static_reset('node_access_view_all_nodes');
     $this->assertCacheContext([
-      0 => 'all:0;node_access_test_author:0',
+      0 => 'view.all:0;node_access_test_author:0',
       1 => 'all',
-      2 => 'all',
-      3 => 'all:0;node_access_test_author:3',
+      2 => 'view.all',
+      3 => 'view.all:0;node_access_test_author:3',
     ]);
 
     // Put user noAccessUser (uid 3) in the realm.
     \Drupal::state()->set('node_access_test.no_access_uid', $this->noAccessUser->id());
     drupal_static_reset('node_access_view_all_nodes');
     $this->assertCacheContext([
-      0 => 'all:0;node_access_test_author:0',
+      0 => 'view.all:0;node_access_test_author:0',
       1 => 'all',
-      2 => 'all:0;node_access_test_author:2;node_access_test:8888,8889',
-      3 => 'all',
+      2 => 'view.all:0;node_access_test_author:2;node_access_test:8888,8889',
+      3 => 'view.all',
     ]);
 
     // Uninstall the node_access_test module
     $this->container->get('module_installer')->uninstall(['node_access_test']);
     drupal_static_reset('node_access_view_all_nodes');
     $this->assertCacheContext([
-      0 => 'all',
+      0 => 'view.all',
       1 => 'all',
-      2 => 'all',
-      3 => 'all',
+      2 => 'view.all',
+      3 => 'view.all',
     ]);
   }
 
diff --git a/core/modules/node/src/Tests/NodeCacheTagsTest.php b/core/modules/node/src/Tests/NodeCacheTagsTest.php
index c18f15b3079a99cabc5282f8c649d12041abe63e..8144ba283bb7a85a5969050f52804d34a01d0e7b 100644
--- a/core/modules/node/src/Tests/NodeCacheTagsTest.php
+++ b/core/modules/node/src/Tests/NodeCacheTagsTest.php
@@ -62,7 +62,7 @@ protected function getAdditionalCacheTagsForEntity(EntityInterface $node) {
    * {@inheritdoc}
    */
   protected function getAdditionalCacheContextsForEntityListing() {
-    return ['node_view_grants'];
+    return ['user.node_grants:view'];
   }
 
 }
diff --git a/core/modules/node/src/Tests/NodeListBuilderTest.php b/core/modules/node/src/Tests/NodeListBuilderTest.php
index a594f1f59fc98e8c8fd8b2813ace107639cd92d9..ec4b1ea89f8de9da77f4b290b6dc4f7534a4829d 100644
--- a/core/modules/node/src/Tests/NodeListBuilderTest.php
+++ b/core/modules/node/src/Tests/NodeListBuilderTest.php
@@ -38,7 +38,7 @@ public function testCacheContexts() {
     $build = $list_builder->render();
     $this->container->get('renderer')->render($build);
 
-    $this->assertEqual(['node_view_grants', 'pager:0'], $build['#cache']['contexts']);
+    $this->assertEqual(['url.query_args.pagers:0', 'user.node_grants:view'], $build['#cache']['contexts']);
   }
 
 }
diff --git a/core/modules/node/src/Tests/Views/FrontPageTest.php b/core/modules/node/src/Tests/Views/FrontPageTest.php
index 61fe5eb170e95e868e60f0645ac647be002f12c2..ed6fd3ddcde9cb41409e765b722d6f9a1cc9b895 100644
--- a/core/modules/node/src/Tests/Views/FrontPageTest.php
+++ b/core/modules/node/src/Tests/Views/FrontPageTest.php
@@ -241,7 +241,7 @@ protected function assertFrontPageViewCacheTags($do_assert_views_caches) {
     $view = Views::getView('frontpage');
     $view->setDisplay('page_1');
 
-    $cache_contexts = ['node_view_grants', 'language'];
+    $cache_contexts = ['user.node_grants:view', 'languages'];
 
     // Test before there are any nodes.
     $empty_node_listing_cache_tags = [
diff --git a/core/modules/system/src/Plugin/Block/SystemMenuBlock.php b/core/modules/system/src/Plugin/Block/SystemMenuBlock.php
index c07fba2480660ba3b8c08fd0b7e7e4feaeec7e82..19eb580d17338f79e004df3864f0ae3c74aac3ac 100644
--- a/core/modules/system/src/Plugin/Block/SystemMenuBlock.php
+++ b/core/modules/system/src/Plugin/Block/SystemMenuBlock.php
@@ -201,7 +201,7 @@ protected function getRequiredCacheContexts() {
     $menu_name = $this->getDerivativeId();
     return [
       'user.roles',
-      'menu.active_trail:' . $menu_name,
+      'route.menu_active_trails:' . $menu_name,
     ];
   }
 
diff --git a/core/modules/system/src/Tests/Cache/PageCacheTagsIntegrationTest.php b/core/modules/system/src/Tests/Cache/PageCacheTagsIntegrationTest.php
index f4d30eceec39b4fe37e6d95558d7f4019a761396..8078913da528afb6d2c052513882bf96c861efc5 100644
--- a/core/modules/system/src/Tests/Cache/PageCacheTagsIntegrationTest.php
+++ b/core/modules/system/src/Tests/Cache/PageCacheTagsIntegrationTest.php
@@ -68,11 +68,11 @@ function testPageCacheTags() {
     ));
 
     $cache_contexts = [
-      'language',
-      'menu.active_trail:account',
-      'menu.active_trail:footer',
-      'menu.active_trail:main',
-      'menu.active_trail:tools',
+      'languages',
+      'route.menu_active_trails:account',
+      'route.menu_active_trails:footer',
+      'route.menu_active_trails:main',
+      'route.menu_active_trails:tools',
       'theme',
       'timezone',
       'user.roles',
diff --git a/core/modules/system/src/Tests/Entity/EntityListBuilderTest.php b/core/modules/system/src/Tests/Entity/EntityListBuilderTest.php
index 9d8b36bd758e5be9e7c7336078ec2d57c270f2a9..fa5097a8bac4aa32fed142e75f4163680cd667e0 100644
--- a/core/modules/system/src/Tests/Entity/EntityListBuilderTest.php
+++ b/core/modules/system/src/Tests/Entity/EntityListBuilderTest.php
@@ -66,7 +66,7 @@ public function testCacheContexts() {
     $build = $list_builder->render();
     $this->container->get('renderer')->render($build);
 
-    $this->assertEqual(['entity_test_view_grants', 'pager:0'], $build['#cache']['contexts']);
+    $this->assertEqual(['entity_test_view_grants', 'url.query_args.pagers:0'], $build['#cache']['contexts']);
   }
 
 }
diff --git a/core/modules/system/tests/modules/pager_test/src/Controller/PagerTestController.php b/core/modules/system/tests/modules/pager_test/src/Controller/PagerTestController.php
index b277c75dbf7a2e41e0668ac2194911567f4f4ca8..2448cce0649b63eebb399409480236eb5ab53d04 100644
--- a/core/modules/system/tests/modules/pager_test/src/Controller/PagerTestController.php
+++ b/core/modules/system/tests/modules/pager_test/src/Controller/PagerTestController.php
@@ -69,7 +69,7 @@ public function queryParameters() {
    * #pre_render callback for #type => pager that shows the pager cache context.
    */
   public static function showPagerCacheContext(array $pager) {
-    drupal_set_message(\Drupal::service('cache_contexts')->convertTokensToKeys(['pager:' . $pager['#element']])[0]);
+    drupal_set_message(\Drupal::service('cache_contexts')->convertTokensToKeys(['url.query_args.pagers:' . $pager['#element']])[0]);
     return $pager;
   }
 
diff --git a/core/modules/taxonomy/src/Plugin/views/argument_default/Tid.php b/core/modules/taxonomy/src/Plugin/views/argument_default/Tid.php
index 5bd3f4570f07482ada1a7ab904c962effb149a31..a8751d57ab52ec13300b9425acc351fff3d7145f 100644
--- a/core/modules/taxonomy/src/Plugin/views/argument_default/Tid.php
+++ b/core/modules/taxonomy/src/Plugin/views/argument_default/Tid.php
@@ -230,7 +230,7 @@ public function isCacheable() {
    * {@inheritdoc}
    */
   public function getCacheContexts() {
-    return ['cache.context.url'];
+    return ['url'];
   }
 
   /**
diff --git a/core/modules/user/src/Cache/UserRolesCacheContext.php b/core/modules/user/src/Cache/UserRolesCacheContext.php
deleted file mode 100644
index 0f7975de46632f0018164476035476d194bd89b7..0000000000000000000000000000000000000000
--- a/core/modules/user/src/Cache/UserRolesCacheContext.php
+++ /dev/null
@@ -1,42 +0,0 @@
-<?php
-
-/**
- * @file
- * Contains \Drupal\user\Cache\UserRolesCacheContext.
- */
-
-namespace Drupal\user\Cache;
-
-use Drupal\Core\Cache\CacheContextInterface;
-use Drupal\Core\Session\AccountInterface;
-
-/**
- * Defines the UserRolesCacheContext service, for "per role" caching.
- */
-class UserRolesCacheContext implements CacheContextInterface {
-
-  /**
-   * Constructs a new UserRolesCacheContext service.
-   *
-   * @param \Drupal\Core\Session\AccountInterface $user
-   *   The current user.
-   */
-  public function __construct(AccountInterface $user) {
-    $this->user = $user;
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public static function getLabel() {
-    return t("User's roles");
-  }
-
-  /**
-   * {@inheritdoc}
-   */
-  public function getContext() {
-    return 'r.' . implode(',', $this->user->getRoles());
-  }
-
-}
diff --git a/core/modules/user/src/Plugin/views/argument_default/User.php b/core/modules/user/src/Plugin/views/argument_default/User.php
index e3d9644f35fc3b9a2caa69ecb63bb581403ad796..269be4e39584daedcffa5fc9311023df5fe7163e 100644
--- a/core/modules/user/src/Plugin/views/argument_default/User.php
+++ b/core/modules/user/src/Plugin/views/argument_default/User.php
@@ -123,7 +123,7 @@ public function isCacheable() {
    * {@inheritdoc}
    */
   public function getCacheContexts() {
-    return ['cache.context.url'];
+    return ['url'];
   }
 
 }
diff --git a/core/modules/user/user.services.yml b/core/modules/user/user.services.yml
index 4bee86981823a50da0bb385425ac87f74b1de126..cb1b741807b3fd423bf0b8223989ac27db643399 100644
--- a/core/modules/user/user.services.yml
+++ b/core/modules/user/user.services.yml
@@ -20,16 +20,6 @@ services:
     arguments: ['@session_configuration']
     tags:
       - { name: authentication_provider, priority: 0 }
-  cache_context.user:
-    class: Drupal\user\Cache\UserCacheContext
-    arguments: ['@current_user']
-    tags:
-      - { name: cache.context}
-  cache_context.user.roles:
-    class: Drupal\user\Cache\UserRolesCacheContext
-    arguments: ['@current_user']
-    tags:
-      - { name: cache.context}
   user.data:
     class: Drupal\user\UserData
     arguments: ['@database']
diff --git a/core/modules/views/src/Entity/View.php b/core/modules/views/src/Entity/View.php
index 51ad1a662b5f5f8dca55e077b45447acd62301ef..24d1ec387100075da1539f5cea6bc87e1fd23e13 100644
--- a/core/modules/views/src/Entity/View.php
+++ b/core/modules/views/src/Entity/View.php
@@ -8,6 +8,7 @@
 namespace Drupal\views\Entity;
 
 use Drupal\Component\Utility\NestedArray;
+use Drupal\Core\Cache\Cache;
 use Drupal\Core\Config\Entity\ConfigEntityBase;
 use Drupal\Core\Entity\EntityStorageInterface;
 use Drupal\views\Views;
@@ -315,10 +316,9 @@ protected function addCacheMetadata() {
       $executable->setDisplay($display_id);
 
       list($display['cache_metadata']['cacheable'], $display['cache_metadata']['contexts']) = $executable->getDisplay()->calculateCacheMetadata();
-      // Always include at least the language context as there will be most
-      // probable translatable strings in the view output.
-      $display['cache_metadata']['contexts'][] = 'language';
-      $display['cache_metadata']['contexts'] = array_unique($display['cache_metadata']['contexts']);
+      // Always include at least the 'languages' context as there will most
+      // probably be translatable strings in the view output.
+      $display['cache_metadata']['contexts'] = Cache::mergeContexts($display['cache_metadata']['contexts'], ['languages']);
     }
     // Restore the previous active display.
     $executable->setDisplay($current_display);
diff --git a/core/modules/views/src/Plugin/views/argument/ArgumentPluginBase.php b/core/modules/views/src/Plugin/views/argument/ArgumentPluginBase.php
index 069276b4df3c41f213f6efb777d57c323e903a0d..4e5469bab77becee1f319977596832409b382a63 100644
--- a/core/modules/views/src/Plugin/views/argument/ArgumentPluginBase.php
+++ b/core/modules/views/src/Plugin/views/argument/ArgumentPluginBase.php
@@ -1208,7 +1208,7 @@ public function getCacheContexts() {
     // By definition arguments depends on the URL.
     // @todo Once contexts are properly injected into block views we could pull
     //   the information from there.
-    $contexts[] = 'cache.context.url';
+    $contexts[] = 'url';
 
     // Asks all subplugins (argument defaults, argument validator and styles).
     if (($plugin = $this->getPlugin('argument_default')) && $plugin instanceof CacheablePluginInterface) {
diff --git a/core/modules/views/src/Plugin/views/argument_default/QueryParameter.php b/core/modules/views/src/Plugin/views/argument_default/QueryParameter.php
index 337b72271dddc3285d95f9917ba178e5317df398..feccd21e1d29068388d4a9103bdfcc64dafe8dab 100644
--- a/core/modules/views/src/Plugin/views/argument_default/QueryParameter.php
+++ b/core/modules/views/src/Plugin/views/argument_default/QueryParameter.php
@@ -95,7 +95,7 @@ public function isCacheable() {
    * {@inheritdoc}
    */
   public function getCacheContexts() {
-    return ['cache.context.url'];
+    return ['url'];
   }
 
 }
diff --git a/core/modules/views/src/Plugin/views/argument_default/Raw.php b/core/modules/views/src/Plugin/views/argument_default/Raw.php
index 4239c50e2f5e5896d062f2412ae73cdbdc6833c4..40bddf3531f2f099ab02b4d9b1ab763c8ec9335d 100644
--- a/core/modules/views/src/Plugin/views/argument_default/Raw.php
+++ b/core/modules/views/src/Plugin/views/argument_default/Raw.php
@@ -124,7 +124,7 @@ public function isCacheable() {
    * {@inheritdoc}
    */
   public function getCacheContexts() {
-    return ['cache.context.url'];
+    return ['url'];
   }
 
 }
diff --git a/core/modules/views/src/Plugin/views/filter/FilterPluginBase.php b/core/modules/views/src/Plugin/views/filter/FilterPluginBase.php
index b5f8bf87549703d2dbeef1bb6cd14763d8aa3a30..9abf63ed109e70c17f98f703b23af2971b11fac6 100644
--- a/core/modules/views/src/Plugin/views/filter/FilterPluginBase.php
+++ b/core/modules/views/src/Plugin/views/filter/FilterPluginBase.php
@@ -1477,7 +1477,7 @@ public function getCacheContexts() {
     // input from GET parameters, which are part of the URL. Hence a view with
     // an exposed filter is cacheable per URL.
     if ($this->isExposed()) {
-      $cache_contexts[] = 'cache.context.url';
+      $cache_contexts[] = 'url';
     }
     return $cache_contexts;
   }
diff --git a/core/modules/views/src/Plugin/views/sort/SortPluginBase.php b/core/modules/views/src/Plugin/views/sort/SortPluginBase.php
index 45b86bee6b0a4ac9f4f2bbddb78f501759ae5d34..8fe2b43b5fc03f9704668f0e3b7f823c1b6c2779 100644
--- a/core/modules/views/src/Plugin/views/sort/SortPluginBase.php
+++ b/core/modules/views/src/Plugin/views/sort/SortPluginBase.php
@@ -241,7 +241,7 @@ public function getCacheContexts() {
     $cache_contexts = [];
     // Exposed sorts use GET parameters, so it depends on the current URL.
     if ($this->isExposed()) {
-      $cache_contexts[] = 'cache.context.url';
+      $cache_contexts[] = 'url';
     }
     return $cache_contexts;
   }
diff --git a/core/modules/views/src/Tests/GlossaryTest.php b/core/modules/views/src/Tests/GlossaryTest.php
index fd979780957bcf283c45b642a6662bbd28dbfb3b..1df4701e6799a5d387fbe8538cbbee6a97138ba4 100644
--- a/core/modules/views/src/Tests/GlossaryTest.php
+++ b/core/modules/views/src/Tests/GlossaryTest.php
@@ -87,7 +87,7 @@ public function testGlossaryView() {
 
     // Verify cache tags.
     $this->enablePageCaching();
-    $this->assertPageCacheContextsAndTags(Url::fromRoute('view.glossary.page_1'), ['cache.context.url', 'node_view_grants', 'language', 'user'], [
+    $this->assertPageCacheContextsAndTags(Url::fromRoute('view.glossary.page_1'), ['languages', 'url', 'user'], [
       'config:views.view.glossary',
       'node:' . $nodes_by_char['a'][0]->id(), 'node:' . $nodes_by_char['a'][1]->id(), 'node:' . $nodes_by_char['a'][2]->id(),
       'node_list',
diff --git a/core/modules/views/src/Tests/RenderCacheIntegrationTest.php b/core/modules/views/src/Tests/RenderCacheIntegrationTest.php
index da94ab4b8ca59af7edb953fcd39e673904841d01..4a9276f17edddd65b462854a2f2eb808adf21d97 100644
--- a/core/modules/views/src/Tests/RenderCacheIntegrationTest.php
+++ b/core/modules/views/src/Tests/RenderCacheIntegrationTest.php
@@ -221,7 +221,7 @@ public function testViewAddCacheMetadata() {
     $view = View::load('test_display');
     $view->save();
 
-    $this->assertEqual(['node_view_grants', 'user', 'language'], $view->getDisplay('default')['cache_metadata']['contexts']);
+    $this->assertEqual(['languages', 'user', 'user.node_grants:view'], $view->getDisplay('default')['cache_metadata']['contexts']);
   }
 
 }
diff --git a/core/tests/Drupal/Tests/Core/Cache/CacheContextsTest.php b/core/tests/Drupal/Tests/Core/Cache/CacheContextsTest.php
index 1563a6df4f4fc1cd2f157183fb65058b0f18dca4..a81f52d4e8c2d1dfc29a41a5a2fa53b43e5351cc 100644
--- a/core/tests/Drupal/Tests/Core/Cache/CacheContextsTest.php
+++ b/core/tests/Drupal/Tests/Core/Cache/CacheContextsTest.php
@@ -19,6 +19,63 @@
  */
 class CacheContextsTest extends UnitTestCase {
 
+  /**
+   * @covers ::optimizeTokens
+   *
+   * @dataProvider providerTestOptimizeTokens
+   */
+  public function testOptimizeTokens(array $context_tokens, array $optimized_context_tokens) {
+    $container = $this->getMockBuilder('Drupal\Core\DependencyInjection\Container')
+      ->disableOriginalConstructor()
+      ->getMock();
+    $container->expects($this->any())
+      ->method('get')
+      ->will($this->returnValueMap([
+        ['a', Container::EXCEPTION_ON_INVALID_REFERENCE, new FooCacheContext()],
+        ['a.b', Container::EXCEPTION_ON_INVALID_REFERENCE, new FooCacheContext()],
+        ['a.b.c', Container::EXCEPTION_ON_INVALID_REFERENCE, new BazCacheContext()],
+        ['x', Container::EXCEPTION_ON_INVALID_REFERENCE, new BazCacheContext()],
+      ]));
+    $cache_contexts = new CacheContexts($container, $this->getContextsFixture());
+
+    $this->assertSame($optimized_context_tokens, $cache_contexts->optimizeTokens($context_tokens));
+  }
+
+  /**
+   * Provides a list of context token sets.
+   */
+  public function providerTestOptimizeTokens() {
+    return [
+      [['a', 'x'], ['a', 'x']],
+      [['a.b', 'x'], ['a.b', 'x']],
+
+      // Direct ancestor, single-level hierarchy.
+      [['a', 'a.b'], ['a']],
+      [['a.b', 'a'], ['a']],
+
+      // Direct ancestor, multi-level hierarchy.
+      [['a.b', 'a.b.c'], ['a.b']],
+      [['a.b.c', 'a.b'], ['a.b']],
+
+      // Indirect ancestor.
+      [['a', 'a.b.c'], ['a']],
+      [['a.b.c', 'a'], ['a']],
+
+      // Direct & indirect ancestors.
+      [['a', 'a.b', 'a.b.c'], ['a']],
+      [['a', 'a.b.c', 'a.b'], ['a']],
+      [['a.b', 'a', 'a.b.c'], ['a']],
+      [['a.b', 'a.b.c', 'a'], ['a']],
+      [['a.b.c', 'a.b', 'a'], ['a']],
+      [['a.b.c', 'a', 'a.b'], ['a']],
+
+      // Using parameters.
+      [['a', 'a.b.c:foo'], ['a']],
+      [['a.b.c:foo', 'a'], ['a']],
+      [['a.b.c:foo', 'a.b.c'], ['a.b.c']],
+    ];
+  }
+
   /**
    * @covers ::convertTokensToKeys
    */
@@ -146,7 +203,7 @@ public static function getLabel() {
   /**
    * {@inheritdoc}
    */
-  public function getContext($parameter) {
+  public function getContext($parameter = NULL) {
     if (!is_string($parameter) || strlen($parameter) ===  0) {
       throw new \Exception();
     }