diff --git a/composer.lock b/composer.lock
index 06555cc03665309c910c79454b71aff187fe9f05..2e4118d494a5620e6f190fdca6e12d06fffbe3aa 100644
--- a/composer.lock
+++ b/composer.lock
@@ -2958,16 +2958,16 @@
         },
         {
             "name": "drupal/core",
-            "version": "9.2.7",
+            "version": "9.2.8",
             "source": {
                 "type": "git",
                 "url": "https://github.com/drupal/core.git",
-                "reference": "ce3220458c7a744bb00e9436e48d8e644e134576"
+                "reference": "b40ae04a0a8a8a0952e77da7545de5ee8fa40757"
             },
             "dist": {
                 "type": "zip",
-                "url": "https://api.github.com/repos/drupal/core/zipball/ce3220458c7a744bb00e9436e48d8e644e134576",
-                "reference": "ce3220458c7a744bb00e9436e48d8e644e134576",
+                "url": "https://api.github.com/repos/drupal/core/zipball/b40ae04a0a8a8a0952e77da7545de5ee8fa40757",
+                "reference": "b40ae04a0a8a8a0952e77da7545de5ee8fa40757",
                 "shasum": ""
             },
             "require": {
@@ -3206,9 +3206,9 @@
             ],
             "description": "Drupal is an open source content management platform powering millions of websites and applications.",
             "support": {
-                "source": "https://github.com/drupal/core/tree/9.2.7"
+                "source": "https://github.com/drupal/core/tree/9.2.8"
             },
-            "time": "2021-10-06T10:34:39+00:00"
+            "time": "2021-11-03T17:25:16+00:00"
         },
         {
             "name": "drupal/core-composer-scaffold",
@@ -3262,16 +3262,16 @@
         },
         {
             "name": "drupal/core-recommended",
-            "version": "9.2.7",
+            "version": "9.2.8",
             "source": {
                 "type": "git",
                 "url": "https://github.com/drupal/core-recommended.git",
-                "reference": "87c998e8d60d6b2452b21827fb7b16f77d02a38a"
+                "reference": "c77ac58a8ef159065a7d28853e41f758a9cb057d"
             },
             "dist": {
                 "type": "zip",
-                "url": "https://api.github.com/repos/drupal/core-recommended/zipball/87c998e8d60d6b2452b21827fb7b16f77d02a38a",
-                "reference": "87c998e8d60d6b2452b21827fb7b16f77d02a38a",
+                "url": "https://api.github.com/repos/drupal/core-recommended/zipball/c77ac58a8ef159065a7d28853e41f758a9cb057d",
+                "reference": "c77ac58a8ef159065a7d28853e41f758a9cb057d",
                 "shasum": ""
             },
             "require": {
@@ -3280,7 +3280,7 @@
                 "doctrine/annotations": "1.13.1",
                 "doctrine/lexer": "1.2.1",
                 "doctrine/reflection": "1.2.2",
-                "drupal/core": "9.2.7",
+                "drupal/core": "9.2.8",
                 "egulias/email-validator": "2.1.25",
                 "guzzlehttp/guzzle": "6.5.5",
                 "guzzlehttp/promises": "1.4.1",
@@ -3343,9 +3343,9 @@
             ],
             "description": "Locked core dependencies; require this project INSTEAD OF drupal/core.",
             "support": {
-                "source": "https://github.com/drupal/core-recommended/tree/9.2.7"
+                "source": "https://github.com/drupal/core-recommended/tree/9.2.8"
             },
-            "time": "2021-10-06T10:34:39+00:00"
+            "time": "2021-11-03T17:25:16+00:00"
         },
         {
             "name": "drupal/crop",
diff --git a/vendor/composer/installed.json b/vendor/composer/installed.json
index f2c3cba795c6794dae580de3c484c271636a68e1..c3d3fbae53f5db3b3205afd9251efd7cf7131e8f 100644
--- a/vendor/composer/installed.json
+++ b/vendor/composer/installed.json
@@ -3039,17 +3039,17 @@
         },
         {
             "name": "drupal/core",
-            "version": "9.2.7",
-            "version_normalized": "9.2.7.0",
+            "version": "9.2.8",
+            "version_normalized": "9.2.8.0",
             "source": {
                 "type": "git",
                 "url": "https://github.com/drupal/core.git",
-                "reference": "ce3220458c7a744bb00e9436e48d8e644e134576"
+                "reference": "b40ae04a0a8a8a0952e77da7545de5ee8fa40757"
             },
             "dist": {
                 "type": "zip",
-                "url": "https://api.github.com/repos/drupal/core/zipball/ce3220458c7a744bb00e9436e48d8e644e134576",
-                "reference": "ce3220458c7a744bb00e9436e48d8e644e134576",
+                "url": "https://api.github.com/repos/drupal/core/zipball/b40ae04a0a8a8a0952e77da7545de5ee8fa40757",
+                "reference": "b40ae04a0a8a8a0952e77da7545de5ee8fa40757",
                 "shasum": ""
             },
             "require": {
@@ -3214,7 +3214,7 @@
                 "drupal/workflows": "self.version",
                 "drupal/workspaces": "self.version"
             },
-            "time": "2021-10-06T10:34:39+00:00",
+            "time": "2021-11-03T17:25:16+00:00",
             "type": "drupal-core",
             "extra": {
                 "drupal-scaffold": {
@@ -3294,7 +3294,7 @@
             ],
             "description": "Drupal is an open source content management platform powering millions of websites and applications.",
             "support": {
-                "source": "https://github.com/drupal/core/tree/9.2.7"
+                "source": "https://github.com/drupal/core/tree/9.2.8"
             },
             "install-path": "../../web/core"
         },
@@ -3350,17 +3350,17 @@
         },
         {
             "name": "drupal/core-recommended",
-            "version": "9.2.7",
-            "version_normalized": "9.2.7.0",
+            "version": "9.2.8",
+            "version_normalized": "9.2.8.0",
             "source": {
                 "type": "git",
                 "url": "https://github.com/drupal/core-recommended.git",
-                "reference": "87c998e8d60d6b2452b21827fb7b16f77d02a38a"
+                "reference": "c77ac58a8ef159065a7d28853e41f758a9cb057d"
             },
             "dist": {
                 "type": "zip",
-                "url": "https://api.github.com/repos/drupal/core-recommended/zipball/87c998e8d60d6b2452b21827fb7b16f77d02a38a",
-                "reference": "87c998e8d60d6b2452b21827fb7b16f77d02a38a",
+                "url": "https://api.github.com/repos/drupal/core-recommended/zipball/c77ac58a8ef159065a7d28853e41f758a9cb057d",
+                "reference": "c77ac58a8ef159065a7d28853e41f758a9cb057d",
                 "shasum": ""
             },
             "require": {
@@ -3369,7 +3369,7 @@
                 "doctrine/annotations": "1.13.1",
                 "doctrine/lexer": "1.2.1",
                 "doctrine/reflection": "1.2.2",
-                "drupal/core": "9.2.7",
+                "drupal/core": "9.2.8",
                 "egulias/email-validator": "2.1.25",
                 "guzzlehttp/guzzle": "6.5.5",
                 "guzzlehttp/promises": "1.4.1",
@@ -3425,7 +3425,7 @@
             "conflict": {
                 "webflo/drupal-core-strict": "*"
             },
-            "time": "2021-10-06T10:34:39+00:00",
+            "time": "2021-11-03T17:25:16+00:00",
             "type": "metapackage",
             "notification-url": "https://packagist.org/downloads/",
             "license": [
@@ -3433,7 +3433,7 @@
             ],
             "description": "Locked core dependencies; require this project INSTEAD OF drupal/core.",
             "support": {
-                "source": "https://github.com/drupal/core-recommended/tree/9.2.7"
+                "source": "https://github.com/drupal/core-recommended/tree/9.2.8"
             },
             "install-path": null
         },
diff --git a/vendor/composer/installed.php b/vendor/composer/installed.php
index 1e63755e084feeb5a42af11b20d70a011fb5cf3b..ddc65fc6a3bbd8b9aa0b61b9f468a5381a77d3f8 100644
--- a/vendor/composer/installed.php
+++ b/vendor/composer/installed.php
@@ -5,7 +5,7 @@
         'type' => 'project',
         'install_path' => __DIR__ . '/../../',
         'aliases' => array(),
-        'reference' => '5baa0f240fa9db4ac35dde696ec3fe32e6e3eec2',
+        'reference' => 'df61ac4c80357a6ff9e6e56957f55630f42d3212',
         'name' => 'osu-asc-webservices/d8-upstream',
         'dev' => true,
     ),
@@ -271,7 +271,7 @@
         'drupal/action' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/addtocalendar' => array(
@@ -304,7 +304,7 @@
         'drupal/aggregator' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/allowed_formats' => array(
@@ -328,25 +328,25 @@
         'drupal/automated_cron' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/ban' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/bartik' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/basic_auth' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/better_exposed_filters' => array(
@@ -361,19 +361,19 @@
         'drupal/big_pipe' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/block' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/block_content' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/block_field' => array(
@@ -406,7 +406,7 @@
         'drupal/book' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/bootstrap' => array(
@@ -421,7 +421,7 @@
         'drupal/breakpoint' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/cache_control_override' => array(
@@ -445,7 +445,7 @@
         'drupal/ckeditor' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/ckeditor_indentblock' => array(
@@ -460,31 +460,31 @@
         'drupal/claro' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/classy' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/color' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/comment' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/config' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/config_direct_save' => array(
@@ -517,7 +517,7 @@
         'drupal/config_translation' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/config_update' => array(
@@ -568,7 +568,7 @@
         'drupal/contact' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/content_access' => array(
@@ -583,52 +583,52 @@
         'drupal/content_moderation' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/content_translation' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/contextual' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core' => array(
-            'pretty_version' => '9.2.7',
-            'version' => '9.2.7.0',
+            'pretty_version' => '9.2.8',
+            'version' => '9.2.8.0',
             'type' => 'drupal-core',
             'install_path' => __DIR__ . '/../../web/core',
             'aliases' => array(),
-            'reference' => 'ce3220458c7a744bb00e9436e48d8e644e134576',
+            'reference' => 'b40ae04a0a8a8a0952e77da7545de5ee8fa40757',
             'dev_requirement' => false,
         ),
         'drupal/core-annotation' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-assertion' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-bridge' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-class-finder' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-composer-scaffold' => array(
@@ -643,136 +643,136 @@
         'drupal/core-datetime' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-dependency-injection' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-diff' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-discovery' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-event-dispatcher' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-file-cache' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-file-security' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-filesystem' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-front-matter' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-gettext' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-graph' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-http-foundation' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-php-storage' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-plugin' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-proxy-builder' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-recommended' => array(
-            'pretty_version' => '9.2.7',
-            'version' => '9.2.7.0',
+            'pretty_version' => '9.2.8',
+            'version' => '9.2.8.0',
             'type' => 'metapackage',
             'install_path' => NULL,
             'aliases' => array(),
-            'reference' => '87c998e8d60d6b2452b21827fb7b16f77d02a38a',
+            'reference' => 'c77ac58a8ef159065a7d28853e41f758a9cb057d',
             'dev_requirement' => false,
         ),
         'drupal/core-render' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-serialization' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-transliteration' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-utility' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-uuid' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/core-version' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/crop' => array(
@@ -796,19 +796,19 @@
         'drupal/datetime' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/datetime_range' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/dblog' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/dropzonejs' => array(
@@ -832,13 +832,13 @@
         'drupal/dynamic_page_cache' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/editor' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/editor_advanced_link' => array(
@@ -907,7 +907,7 @@
         'drupal/entity_reference' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/entity_reference_revisions' => array(
@@ -931,7 +931,7 @@
         'drupal/field' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/field_group' => array(
@@ -946,7 +946,7 @@
         'drupal/field_layout' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/field_permissions' => array(
@@ -961,13 +961,13 @@
         'drupal/field_ui' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/file' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/file_browser' => array(
@@ -982,7 +982,7 @@
         'drupal/filter' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/focal_point' => array(
@@ -997,7 +997,7 @@
         'drupal/forum' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/google_analytics' => array(
@@ -1021,25 +1021,25 @@
         'drupal/hal' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/help' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/help_topics' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/history' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/honeypot' => array(
@@ -1054,7 +1054,7 @@
         'drupal/image' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/inline_entity_form' => array(
@@ -1069,7 +1069,7 @@
         'drupal/inline_form_errors' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/jquery_ui' => array(
@@ -1111,25 +1111,25 @@
         'drupal/jsonapi' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/language' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/layout_builder' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/layout_discovery' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/libraries' => array(
@@ -1144,7 +1144,7 @@
         'drupal/link' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/link_attributes' => array(
@@ -1168,7 +1168,7 @@
         'drupal/locale' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/mathjax' => array(
@@ -1183,7 +1183,7 @@
         'drupal/media' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/media_entity_browser' => array(
@@ -1207,7 +1207,7 @@
         'drupal/media_library' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/menu_block' => array(
@@ -1240,13 +1240,13 @@
         'drupal/menu_link_content' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/menu_ui' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/metatag' => array(
@@ -1261,7 +1261,7 @@
         'drupal/migrate' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/migrate_devel' => array(
@@ -1276,19 +1276,19 @@
         'drupal/migrate_drupal' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/migrate_drupal_multilingual' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/migrate_drupal_ui' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/migrate_plus' => array(
@@ -1312,7 +1312,7 @@
         'drupal/minimal' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/mobile_detect' => array(
@@ -1354,25 +1354,25 @@
         'drupal/node' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/olivero' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/options' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/page_cache' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/pantheon_advanced_page_cache' => array(
@@ -1396,13 +1396,13 @@
         'drupal/path' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/path_alias' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/pathauto' => array(
@@ -1426,13 +1426,13 @@
         'drupal/quickedit' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/rdf' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/rebuild_cache_access' => array(
@@ -1474,13 +1474,13 @@
         'drupal/responsive_image' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/rest' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/roleassign' => array(
@@ -1504,7 +1504,7 @@
         'drupal/search' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/search_api' => array(
@@ -1528,25 +1528,25 @@
         'drupal/serialization' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/settings_tray' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/seven' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/shortcut' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/simple_gmap' => array(
@@ -1615,19 +1615,19 @@
         'drupal/standard' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/stark' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/statistics' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/superfish' => array(
@@ -1651,31 +1651,31 @@
         'drupal/syslog' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/system' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/taxonomy' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/telephone' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/text' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/token' => array(
@@ -1690,19 +1690,19 @@
         'drupal/toolbar' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/tour' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/tracker' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/twig_tweak' => array(
@@ -1735,13 +1735,13 @@
         'drupal/update' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/user' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/userprotect' => array(
@@ -1774,7 +1774,7 @@
         'drupal/views' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/views_ajax_history' => array(
@@ -1834,7 +1834,7 @@
         'drupal/views_ui' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/webform' => array(
@@ -1849,13 +1849,13 @@
         'drupal/workflows' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drupal/workspaces' => array(
             'dev_requirement' => false,
             'replaced' => array(
-                0 => '9.2.7',
+                0 => '9.2.8',
             ),
         ),
         'drush-ops/behat-drush-endpoint' => array(
@@ -2104,7 +2104,7 @@
             'type' => 'project',
             'install_path' => __DIR__ . '/../../',
             'aliases' => array(),
-            'reference' => '5baa0f240fa9db4ac35dde696ec3fe32e6e3eec2',
+            'reference' => 'df61ac4c80357a6ff9e6e56957f55630f42d3212',
             'dev_requirement' => false,
         ),
         'pantheon-systems/quicksilver-pushback' => array(
diff --git a/web/core/lib/Drupal.php b/web/core/lib/Drupal.php
index 286a13668d9ba9179ed253c2404502cdf6778375..b4b1be906e0db56ac269cc5e4895f1010f05437b 100644
--- a/web/core/lib/Drupal.php
+++ b/web/core/lib/Drupal.php
@@ -75,7 +75,7 @@ class Drupal {
   /**
    * The current system version.
    */
-  const VERSION = '9.2.7';
+  const VERSION = '9.2.8';
 
   /**
    * Core API compatibility.
diff --git a/web/core/lib/Drupal/Core/Asset/CssCollectionOptimizer.php b/web/core/lib/Drupal/Core/Asset/CssCollectionOptimizer.php
index 62dff2c7f92aec3f2a1faab1a563175677547404..7d47dad86e6ce688188f5f9a24647c06f9733524 100644
--- a/web/core/lib/Drupal/Core/Asset/CssCollectionOptimizer.php
+++ b/web/core/lib/Drupal/Core/Asset/CssCollectionOptimizer.php
@@ -123,8 +123,12 @@ public function optimize(array $css_assets) {
               // Per the W3C specification at
               // http://www.w3.org/TR/REC-CSS2/cascade.html#at-import, @import
               // rules must precede any other style, so we move those to the
-              // top.
-              $regexp = '/@import[^;]+;/i';
+              // top. The regular expression is expressed in NOWDOC since it is
+              // detecting backslashes as well as single and double quotes. It
+              // is difficult to read when represented as a quoted string.
+              $regexp = <<<'REGEXP'
+/@import\s*(?:'(?:\\'|.)*'|"(?:\\"|.)*"|url\(\s*(?:\\[\)\'\"]|[^'")])*\s*\)|url\(\s*'(?:\'|.)*'\s*\)|url\(\s*"(?:\"|.)*"\s*\)).*;/iU
+REGEXP;
               preg_match_all($regexp, $data, $matches);
               $data = preg_replace($regexp, '', $data);
               $data = implode('', $matches[0]) . $data;
diff --git a/web/core/lib/Drupal/Core/Asset/CssOptimizer.php b/web/core/lib/Drupal/Core/Asset/CssOptimizer.php
index e939b3d0542f16fcb700dfc36b0b24fed1a3d45c..0aec443f987e0703d11f3678fb1d08b8466c8291 100644
--- a/web/core/lib/Drupal/Core/Asset/CssOptimizer.php
+++ b/web/core/lib/Drupal/Core/Asset/CssOptimizer.php
@@ -166,7 +166,7 @@ protected function loadNestedFile($matches) {
     // the url() path.
     $directory = $directory == '.' ? '' : $directory . '/';
 
-    // Alter all internal url() paths. Leave external paths alone. We don't need
+    // Alter all internal asset paths. Leave external paths alone. We don't need
     // to normalize absolute paths here because that will be done later.
     return preg_replace('/url\(\s*([\'"]?)(?![a-z]+:|\/+)([^\'")]+)([\'"]?)\s*\)/i', 'url(\1' . $directory . '\2\3)', $file);
   }
@@ -230,7 +230,9 @@ protected function processCss($contents, $optimize = FALSE) {
     }
 
     // Replaces @import commands with the actual stylesheet content.
-    // This happens recursively but omits external files.
+    // This happens recursively but omits external files and local files
+    // with supports- or media-query qualifiers, as those are conditionally
+    // loaded depending on the user agent.
     $contents = preg_replace_callback('/@import\s*(?:url\(\s*)?[\'"]?(?![a-z]+:)(?!\/\/)([^\'"\()]+)[\'"]?\s*\)?\s*;/', [$this, 'loadNestedFile'], $contents);
 
     return $contents;
diff --git a/web/core/lib/Drupal/Core/Logger/LogMessageParser.php b/web/core/lib/Drupal/Core/Logger/LogMessageParser.php
index 690847e7d5556ad66225059109d60f174085b188..dc0ba4fdc5ffeace1bd4c32b39d90e5162ab5c82 100644
--- a/web/core/lib/Drupal/Core/Logger/LogMessageParser.php
+++ b/web/core/lib/Drupal/Core/Logger/LogMessageParser.php
@@ -17,7 +17,7 @@ public function parseMessagePlaceholders(&$message, array &$context) {
       $has_psr3 = TRUE;
       // Transform PSR3 style messages containing placeholders to
       // \Drupal\Component\Render\FormattableMarkup style.
-      $message = preg_replace('/\{(.*)\}/U', '@$1', $message);
+      $message = preg_replace('/\{([^\{}]*)\}/U', '@$1', $message);
     }
     foreach ($context as $key => $variable) {
       // PSR3 style placeholders.
diff --git a/web/core/lib/Drupal/Core/Render/Element/MachineName.php b/web/core/lib/Drupal/Core/Render/Element/MachineName.php
index 9e0532646861ca54931f8095ff5ef41760f10075..555a044c0e29f56cfd4db0d896141e179a44d954 100644
--- a/web/core/lib/Drupal/Core/Render/Element/MachineName.php
+++ b/web/core/lib/Drupal/Core/Render/Element/MachineName.php
@@ -44,7 +44,8 @@
  *   - error: (optional) A custom form error message string to show, if the
  *     machine name contains disallowed characters.
  *   - standalone: (optional) Whether the live preview should stay in its own
- *     form element rather than in the suffix of the source element. Defaults
+ *     form element rather than in the suffix of the source element. The source
+ *     element must appear in the form structure before this element. Defaults
  *     to FALSE.
  * - #maxlength: (optional) Maximum allowed length of the machine name. Defaults
  *   to 64.
@@ -183,6 +184,13 @@ public static function processMachineName(&$element, FormStateInterface $form_st
       return $element;
     }
 
+    // The source element must be defined before the machine name element.
+    if (!isset($source['#id'])) {
+      $element_parents = implode('][', $element['#array_parents']);
+      $source_parents = implode('][', $element['#machine_name']['source']);
+      throw new \LogicException(sprintf('The machine name element "%s" is defined before the source element "%s", it must be defined after or the source element must specify an id.', $element_parents, $source_parents));
+    }
+
     $suffix_id = $source['#id'] . '-machine-name-suffix';
     $element['#machine_name']['suffix'] = '#' . $suffix_id;
 
diff --git a/web/core/lib/Drupal/Core/Render/HtmlResponseAttachmentsProcessor.php b/web/core/lib/Drupal/Core/Render/HtmlResponseAttachmentsProcessor.php
index d698604b15bf9bfc6716b3fabb7ee6259a5a7e39..2dc416f4bd7b19a8a9b55c0b35efc58c851ac539 100644
--- a/web/core/lib/Drupal/Core/Render/HtmlResponseAttachmentsProcessor.php
+++ b/web/core/lib/Drupal/Core/Render/HtmlResponseAttachmentsProcessor.php
@@ -436,7 +436,15 @@ protected function processHtmlHeadLink(array $html_head_link) {
         '#attributes' => $attributes,
       ];
       $href = $attributes['href'];
-      $attached['html_head'][] = [$element, 'html_head_link:' . $attributes['rel'] . ':' . $href];
+      $rel = $attributes['rel'];
+
+      // Allow multiple hreflang tags to use the same href.
+      if (isset($attributes['hreflang'])) {
+        $attached['html_head'][] = [$element, 'html_head_link:' . $rel . ':' . $attributes['hreflang'] . ':' . $href];
+      }
+      else {
+        $attached['html_head'][] = [$element, 'html_head_link:' . $rel . ':' . $href];
+      }
 
       if ($should_add_header) {
         // Also add a HTTP header "Link:".
diff --git a/web/core/lib/Drupal/Core/Test/TestRunnerKernel.php b/web/core/lib/Drupal/Core/Test/TestRunnerKernel.php
index 85d84b75a7bd0f2ee72a94e5bd979bdf906c37a1..2315693eacf442ce5d7c5d44a8f13a73a688d7d3 100644
--- a/web/core/lib/Drupal/Core/Test/TestRunnerKernel.php
+++ b/web/core/lib/Drupal/Core/Test/TestRunnerKernel.php
@@ -87,8 +87,8 @@ public function boot() {
 
     // Create the build/artifacts directory if necessary.
     include_once $this->getAppRoot() . '/core/includes/file.inc';
-    if (!is_dir('public://simpletest')) {
-      mkdir('public://simpletest', 0777, TRUE);
+    if (!is_dir('public://simpletest') && !@mkdir('public://simpletest', 0777, TRUE) && !is_dir('public://simpletest')) {
+      throw new \RuntimeException('Unable to create directory: public://simpletest');
     }
   }
 
diff --git a/web/core/misc/cspell/dictionary.txt b/web/core/misc/cspell/dictionary.txt
index 80d6cb2fc6795b4bea1a56410f88a70899180164..9a23b6260e4df9ddd922147c7b5e5df354c0c800 100644
--- a/web/core/misc/cspell/dictionary.txt
+++ b/web/core/misc/cspell/dictionary.txt
@@ -1045,6 +1045,7 @@ nothere
 notnull
 notsimpletest
 nourriture
+nowdoc
 nplurals
 nresponse
 ntfs
diff --git a/web/core/modules/block/src/Plugin/migrate/source/Block.php b/web/core/modules/block/src/Plugin/migrate/source/Block.php
index 20b128c3a06170653a9dded4862cdb3fab214c2f..16fd6cd9aa251e493c393231971dc4b71de7b07a 100644
--- a/web/core/modules/block/src/Plugin/migrate/source/Block.php
+++ b/web/core/modules/block/src/Plugin/migrate/source/Block.php
@@ -6,9 +6,10 @@
 use Drupal\migrate_drupal\Plugin\migrate\source\DrupalSqlBase;
 
 /**
- * Drupal block source from database.
+ * Drupal 6/7 block source from database.
+ *
+ * For available configuration keys, refer to the parent classes.
  *
- * For available configuration keys, refer to the parent classes:
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/block/src/Plugin/migrate/source/d6/BlockTranslation.php b/web/core/modules/block/src/Plugin/migrate/source/d6/BlockTranslation.php
index 007c25d28946778fc0d06e9e0019920d320e3bc4..6cfa1d6ffcd6041fc67d9159f3dbd5587717344f 100644
--- a/web/core/modules/block/src/Plugin/migrate/source/d6/BlockTranslation.php
+++ b/web/core/modules/block/src/Plugin/migrate/source/d6/BlockTranslation.php
@@ -7,9 +7,10 @@
 use Drupal\migrate\Row;
 
 /**
- * Gets i18n block data from source database.
+ * Drupal 6 i18n block data from database.
+ *
+ * For available configuration keys, refer to the parent classes.
  *
- * For available configuration keys, refer to the parent classes:
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/block/src/Plugin/migrate/source/d7/BlockTranslation.php b/web/core/modules/block/src/Plugin/migrate/source/d7/BlockTranslation.php
index 132599357b6d6ef90611713e62d9e2e38b6a6cf8..8eb01c8015ccb515dd5cbfe3dec9cab95ae8072e 100644
--- a/web/core/modules/block/src/Plugin/migrate/source/d7/BlockTranslation.php
+++ b/web/core/modules/block/src/Plugin/migrate/source/d7/BlockTranslation.php
@@ -5,9 +5,10 @@
 use Drupal\block\Plugin\migrate\source\Block;
 
 /**
- * Gets i18n block data from source database.
+ * Drupal 7 i18n block data from database.
+ *
+ * For available configuration keys, refer to the parent classes.
  *
- * For available configuration keys, refer to the parent classes:
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/block_content/src/Plugin/migrate/source/d6/Box.php b/web/core/modules/block_content/src/Plugin/migrate/source/d6/Box.php
index 1e8a88d471ee89d83f0b2914effb19a4467d57a0..f264f20892277b66172fd4f6ae204091e67680be 100644
--- a/web/core/modules/block_content/src/Plugin/migrate/source/d6/Box.php
+++ b/web/core/modules/block_content/src/Plugin/migrate/source/d6/Box.php
@@ -7,7 +7,8 @@
 /**
  * Drupal 6 block source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/block_content/src/Plugin/migrate/source/d6/BoxTranslation.php b/web/core/modules/block_content/src/Plugin/migrate/source/d6/BoxTranslation.php
index 50685e3b3dad4dfb5211562f5f33d5abeaace9fb..234af073c01bd900fe3a3c86a2bb10fd499e80c3 100644
--- a/web/core/modules/block_content/src/Plugin/migrate/source/d6/BoxTranslation.php
+++ b/web/core/modules/block_content/src/Plugin/migrate/source/d6/BoxTranslation.php
@@ -5,9 +5,10 @@
 use Drupal\block_content\Plugin\migrate\source\d7\BlockCustomTranslation as D7BlockCustomTranslation;
 
 /**
- * Gets Drupal 6 i18n custom block translations from database.
+ * Drupal 6 i18n custom block translations source from database.
+ *
+ * For available configuration keys, refer to the parent classes.
  *
- * For available configuration keys, refer to the parent classes:
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/block_content/src/Plugin/migrate/source/d7/BlockCustom.php b/web/core/modules/block_content/src/Plugin/migrate/source/d7/BlockCustom.php
index 4b9c5e89cca8345ffa856491c6c979b6d59d56fe..70a0711f4030c95d67660f15831019b175d17155 100644
--- a/web/core/modules/block_content/src/Plugin/migrate/source/d7/BlockCustom.php
+++ b/web/core/modules/block_content/src/Plugin/migrate/source/d7/BlockCustom.php
@@ -7,7 +7,8 @@
 /**
  * Drupal 7 custom block source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/block_content/src/Plugin/migrate/source/d7/BlockCustomTranslation.php b/web/core/modules/block_content/src/Plugin/migrate/source/d7/BlockCustomTranslation.php
index a297ff3cea8c662c090767cef1beac9a26d613c6..8298a0ee939efde1c3340667d1b3635e0cbf8a31 100644
--- a/web/core/modules/block_content/src/Plugin/migrate/source/d7/BlockCustomTranslation.php
+++ b/web/core/modules/block_content/src/Plugin/migrate/source/d7/BlockCustomTranslation.php
@@ -7,9 +7,10 @@
 use Drupal\content_translation\Plugin\migrate\source\I18nQueryTrait;
 
 /**
- * Gets Drupal 7 i18n custom block translations from database.
+ * Drupal 7 i18n custom block translations source from database.
+ *
+ * For available configuration keys, refer to the parent classes.
  *
- * For available configuration keys, refer to the parent classes:
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/ckeditor/src/Plugin/CKEditorPlugin/Internal.php b/web/core/modules/ckeditor/src/Plugin/CKEditorPlugin/Internal.php
index 8b4dbe098013ad70fd6fb172b2dc92229016236d..135cdee2902f0b9dd4c1a90c2354f55f4f86b80a 100644
--- a/web/core/modules/ckeditor/src/Plugin/CKEditorPlugin/Internal.php
+++ b/web/core/modules/ckeditor/src/Plugin/CKEditorPlugin/Internal.php
@@ -116,8 +116,8 @@ public function getConfig(Editor $editor) {
 
     // Add the format_tags setting, if its button is enabled.
     $toolbar_buttons = CKEditorPluginManager::getEnabledButtons($editor);
-    if (in_array('Format', $toolbar_buttons)) {
-      $config['format_tags'] = $this->generateFormatTagsSetting($editor);
+    if (in_array('Format', $toolbar_buttons) && $format_string = $this->generateFormatTagsSetting($editor)) {
+      $config['format_tags'] = $format_string;
     }
 
     return $config;
@@ -344,14 +344,15 @@ public function getButtons() {
    * @param \Drupal\editor\Entity\Editor $editor
    *   A configured text editor object.
    *
-   * @return array
-   *   An array containing the "format_tags" configuration.
+   * @return string|false
+   *   A string containing the "format_tags" configuration or FALSE if the
+   *   editor has not an associated filter format.
    */
   protected function generateFormatTagsSetting(Editor $editor) {
     // When no text format is associated yet, assume no tag is allowed.
     // @see \Drupal\editor\EditorInterface::hasAssociatedFilterFormat()
     if (!$editor->hasAssociatedFilterFormat()) {
-      return [];
+      return FALSE;
     }
 
     $format = $editor->getFilterFormat();
diff --git a/web/core/modules/config_translation/src/Plugin/migrate/source/d6/ProfileFieldTranslation.php b/web/core/modules/config_translation/src/Plugin/migrate/source/d6/ProfileFieldTranslation.php
index 2ccb43cb6106dbf6a8d9ad08f5006c9057c588fe..11215b4403eeed58fe0715a4347515e9ae1ece54 100644
--- a/web/core/modules/config_translation/src/Plugin/migrate/source/d6/ProfileFieldTranslation.php
+++ b/web/core/modules/config_translation/src/Plugin/migrate/source/d6/ProfileFieldTranslation.php
@@ -7,7 +7,8 @@
 /**
  * Drupal 6 i18n strings profile field source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/content_translation/migrations/d7_language_content_comment_settings.yml b/web/core/modules/content_translation/migrations/d7_language_content_comment_settings.yml
index c40173d4fca9f38676cb7c60827e07bca50cd7f5..4515535d13242b6d80cffc1b72fff1c7af2287d2 100644
--- a/web/core/modules/content_translation/migrations/d7_language_content_comment_settings.yml
+++ b/web/core/modules/content_translation/migrations/d7_language_content_comment_settings.yml
@@ -8,22 +8,17 @@ source:
   plugin: d7_language_content_settings
   constants:
     target_type: comment
-    id_prefix: comment_node_
-    label_suffix: comment
 process:
   target_bundle:
     -
-      plugin: concat
+      plugin: migration_lookup
+      migration: d7_comment_type
+      no_stub: true
       source:
-        - constants/id_prefix
         - type
     -
-      plugin: static_map
-      bypass: true
-      # The Forum module provides its own comment type (comment_forum), which we
-      # want to reuse if it exists.
-      map:
-        comment_node_forum: comment_forum
+      plugin: skip_on_empty
+      method: row
   target_entity_type_id: constants/target_type
   default_langcode:
     -
diff --git a/web/core/modules/content_translation/migrations/d7_taxonomy_term_translation.yml b/web/core/modules/content_translation/migrations/d7_taxonomy_term_translation.yml
index a964a26f3117d4f82abb45fdaf0e61c8c3258aa5..10091a8e7607744fc7a3d6169afafd94974f212b 100644
--- a/web/core/modules/content_translation/migrations/d7_taxonomy_term_translation.yml
+++ b/web/core/modules/content_translation/migrations/d7_taxonomy_term_translation.yml
@@ -8,18 +8,6 @@ source:
   plugin: d7_taxonomy_term_translation
   translations: true
 process:
-  skip:
-    -
-      plugin: static_map
-      source: i18n_mode
-      default_value: 0
-      map:
-        1: 0
-        2: 2
-        4: 4
-    -
-      plugin: skip_on_empty
-      method: row
   # If you are using this file to build a custom migration consider removing
   # the tid field to allow incremental migrations.
   tid: tid
diff --git a/web/core/modules/dblog/tests/src/Functional/DbLogTest.php b/web/core/modules/dblog/tests/src/Functional/DbLogTest.php
index e3b7931519ae0760f2ece83724b6591dcb4470c8..1e57cde952b69f3ab99ca101ea1ade89d7e0f898 100644
--- a/web/core/modules/dblog/tests/src/Functional/DbLogTest.php
+++ b/web/core/modules/dblog/tests/src/Functional/DbLogTest.php
@@ -245,6 +245,23 @@ public function testLogEventPageWithMissingInfo() {
     $this->assertSession()->linkNotExists($request_uri);
   }
 
+  /**
+   * Test that twig errors are displayed correctly.
+   */
+  public function testMessageParsing() {
+    $this->drupalLogin($this->adminUser);
+    // Log a common twig error with {{ }} and { } variables.
+    \Drupal::service('logger.factory')->get("php")
+      ->error('Incorrect parameter {{foo}} in path {path}: {value}',
+        ['foo' => 'bar', 'path' => '/baz', 'value' => 'horse']
+      );
+    // View the log page to verify it's correct.
+    $wid = \Drupal::database()->query('SELECT MAX(wid) FROM {watchdog}')->fetchField();
+    $this->drupalGet('admin/reports/dblog/event/' . $wid);
+    $this->assertSession()
+      ->responseContains('Incorrect parameter {bar} in path /baz: horse');
+  }
+
   /**
    * Verifies setting of the database log row limit.
    *
diff --git a/web/core/modules/editor/editor.module b/web/core/modules/editor/editor.module
index 2dd2ed976028818b9cbad43bee47677eb059e8e8..f9dbec18049dbb404a983d87999952ed2be7b9e1 100644
--- a/web/core/modules/editor/editor.module
+++ b/web/core/modules/editor/editor.module
@@ -396,13 +396,19 @@ function editor_entity_update(EntityInterface $entity) {
   // File references that existed both in the previous version of the revision
   // and in the new one don't need their usage to be updated.
   else {
-    $original_uuids_by_field = _editor_get_file_uuids_by_field($entity->original);
+    $original_uuids_by_field = empty($entity->original) ? [] :
+      _editor_get_file_uuids_by_field($entity->original);
+
     $uuids_by_field = _editor_get_file_uuids_by_field($entity);
 
     // Detect file usages that should be incremented.
     foreach ($uuids_by_field as $field => $uuids) {
-      $added_files = array_diff($uuids_by_field[$field], $original_uuids_by_field[$field]);
-      _editor_record_file_usage($added_files, $entity);
+      $original_uuids = isset($original_uuids_by_field[$field]) ?
+        $original_uuids_by_field[$field] : [];
+
+      if ($added_files = array_diff($uuids_by_field[$field], $original_uuids)) {
+        _editor_record_file_usage($added_files, $entity);
+      }
     }
 
     // Detect file usages that should be decremented.
diff --git a/web/core/modules/editor/tests/modules/editor_test.module b/web/core/modules/editor/tests/modules/editor_test.module
index 70fe59fa4151a984bc6f688e712cd3ddd4d79547..bdb7292d62a83ac1801728de8c8fc14121ba2a93 100644
--- a/web/core/modules/editor/tests/modules/editor_test.module
+++ b/web/core/modules/editor/tests/modules/editor_test.module
@@ -5,9 +5,45 @@
  * Helper module for the Text Editor tests.
  */
 
+use Drupal\Core\Entity\EntityInterface;
+use Drupal\node\NodeInterface;
 use Drupal\filter\FilterFormatInterface;
 use Drupal\file\FileInterface;
 
+/**
+ * Implements hook_entity_update().
+ *
+ * @see \Drupal\Tests\editor\Kernel\EntityUpdateTest
+ */
+function editor_test_entity_update(EntityInterface $entity) {
+  // Only act on nodes.
+  if (!$entity instanceof NodeInterface) {
+    return;
+  }
+
+  // Avoid infinite loop by only going through our post save logic once.
+  if (!empty($entity->editor_test_updating)) {
+    return;
+  }
+
+  // Set flag for whether or not the entity needs to be resaved.
+  $needs_update = FALSE;
+
+  // Perform our post save logic.
+  if ($entity->title->value == 'test updated') {
+    // Change the node title.
+    $entity->title->value = 'test updated 2';
+    $needs_update = TRUE;
+  }
+
+  if ($needs_update) {
+    // Set flag on entity that our logic was already executed.
+    $entity->editor_test_updating = TRUE;
+    // And resave entity.
+    $entity->save();
+  }
+}
+
 /**
  * Implements hook_editor_js_settings_alter().
  */
diff --git a/web/core/modules/editor/tests/src/FunctionalJavascript/EditorAdminTest.php b/web/core/modules/editor/tests/src/FunctionalJavascript/EditorAdminTest.php
index 51aee88140011fbbb18408ab848cf6dce8c3b5ce..3d19d3e59961f2d92649e9fdd6b5c0ec5e024e17 100644
--- a/web/core/modules/editor/tests/src/FunctionalJavascript/EditorAdminTest.php
+++ b/web/core/modules/editor/tests/src/FunctionalJavascript/EditorAdminTest.php
@@ -54,6 +54,7 @@ public function testEditorSelection() {
     $this->assertNotEmpty($assert_session->waitForText('sulaco'));
     $page->selectFieldOption('editor[editor]', 'ckeditor');
     $this->assertNotEmpty($this->assertSession()->waitForElementVisible('css', 'ul.ckeditor-toolbar-group-buttons'));
+    $this->assertNotEmpty($this->assertSession()->waitForElementVisible('css', '#ckeditor-plugin-settings'));
     $page->pressButton('Save configuration');
 
     // Test that toggling the editor selection off and back on works.
diff --git a/web/core/modules/editor/tests/src/Kernel/EntityUpdateTest.php b/web/core/modules/editor/tests/src/Kernel/EntityUpdateTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..1a454ea349e432cdb9d07d27fca0ca6e0770a8e6
--- /dev/null
+++ b/web/core/modules/editor/tests/src/Kernel/EntityUpdateTest.php
@@ -0,0 +1,68 @@
+<?php
+
+namespace Drupal\Tests\editor\Kernel;
+
+use Drupal\KernelTests\Core\Entity\EntityKernelTestBase;
+use Drupal\node\Entity\NodeType;
+use Drupal\node\Entity\Node;
+
+/**
+ * Tests updating an entity.
+ *
+ * @group editor
+ */
+class EntityUpdateTest extends EntityKernelTestBase {
+
+  /**
+   * Modules to enable.
+   *
+   * @var array
+   */
+  protected static $modules = ['editor', 'editor_test', 'node'];
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function setUp(): void {
+    parent::setUp();
+    $this->installSchema('node', ['node_access']);
+    $this->installConfig(['node']);
+
+    // Create a node type for testing.
+    $type = NodeType::create(['type' => 'page', 'name' => 'page']);
+    $type->save();
+
+    // Set editor_test module weight to be lower than editor module's weight so
+    // that editor_test_entity_update() is called before editor_entity_update().
+    $extension_config = \Drupal::configFactory()->get('core.extension');
+    $editor_module_weight = $extension_config->get('module.editor');
+    module_set_weight('editor_test', $editor_module_weight - 1);
+  }
+
+  /**
+   * Tests updating an existing entity.
+   *
+   * @see editor_test_entity_update()
+   */
+  public function testEntityUpdate() {
+    // Create a node.
+    $node = Node::create([
+      'type' => 'page',
+      'title' => 'test',
+    ]);
+    $node->save();
+
+    // Update the node.
+    // What happens is the following:
+    // 1. \Drupal\Core\Entity\EntityStorageBase::doPostSave() gets called.
+    // 2. editor_test_entity_update() gets called.
+    // 3. A resave of the updated entity gets triggered (second save call).
+    // 4. \Drupal\Core\Entity\EntityStorageBase::doPostSave() gets called.
+    // 5. editor_test_entity_update() gets called.
+    // 6. editor_entity_update() gets called (caused by the second save call).
+    // 7. editor_entity_update() gets called (caused by the first save call).
+    $node->title->value = 'test updated';
+    $node->save();
+  }
+
+}
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d6/Field.php b/web/core/modules/field/src/Plugin/migrate/source/d6/Field.php
index d2c6392ae241a933d41fb01bf2c0433438b4a651..c3fba78ecccddf019bb894c6d64c96e3350518c4 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d6/Field.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d6/Field.php
@@ -8,7 +8,8 @@
 /**
  * Drupal 6 field source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstance.php b/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstance.php
index ed302ea6dbbacc5de4fc62eb30af359a16b52905..66f24fd2237fd49ae523ba8251cf09d9b322d325 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstance.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstance.php
@@ -26,7 +26,8 @@
  * In this example field instances of type page are retrieved from the source
  * database.
  *
- * For additional configuration keys, refer to the parent classes:
+ * For additional configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstanceOptionTranslation.php b/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstanceOptionTranslation.php
index 505aa1c9aafb73f2986063527ce6b4538ad06eca..c0f58b78e5007959ed226a7624540e794fac3f6f 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstanceOptionTranslation.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstanceOptionTranslation.php
@@ -5,7 +5,8 @@
 /**
  * Drupal 6 i18n field instance option labels source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstancePerFormDisplay.php b/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstancePerFormDisplay.php
index 438b2198e613c8d580c20026936c8da302dc7444..9112e9a63ee39dc64ebd040b964adcab71309ec9 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstancePerFormDisplay.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstancePerFormDisplay.php
@@ -7,7 +7,8 @@
 /**
  * Drupal 6 field instance per form display source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstancePerViewMode.php b/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstancePerViewMode.php
index b571991084bcf0d1a6b1f37ffd720eefafc0ce41..62b4e3d36854c55461c3e725c97a6e3ae6149df7 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstancePerViewMode.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d6/FieldInstancePerViewMode.php
@@ -7,7 +7,8 @@
 /**
  * Drupal 6 field instance per view mode source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d6/FieldLabelDescriptionTranslation.php b/web/core/modules/field/src/Plugin/migrate/source/d6/FieldLabelDescriptionTranslation.php
index 81edff92090ee78e52408797f5608122406df486..51382e6cc6fcd9cda5fad15bf4459b43df4880f2 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d6/FieldLabelDescriptionTranslation.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d6/FieldLabelDescriptionTranslation.php
@@ -7,7 +7,8 @@
 /**
  * Drupal 6 i18n field label and description source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d6/FieldOptionTranslation.php b/web/core/modules/field/src/Plugin/migrate/source/d6/FieldOptionTranslation.php
index 94aaeb43682c5c0f9d35bb7758d9b82c7045dadb..325a27e8baa3e7c9671f5c03c7aa9f4d839fb27c 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d6/FieldOptionTranslation.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d6/FieldOptionTranslation.php
@@ -5,7 +5,8 @@
 /**
  * Drupal 6 i18n field option labels source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d7/Field.php b/web/core/modules/field/src/Plugin/migrate/source/d7/Field.php
index ae71d7dde0ff9a13fd1d5f91b714acb2d201bade..f978146afba9e8b87c7af34882995a9c7ce4d2b6 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d7/Field.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d7/Field.php
@@ -12,7 +12,8 @@
  * migrated. The values of those fields will be migrated to the base fields they
  * were replacing.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d7/FieldInstance.php b/web/core/modules/field/src/Plugin/migrate/source/d7/FieldInstance.php
index d23c1b52c4fc559345f94706decf4a09f79d60e3..079139b688dba3dfb6967ee93e2c9a015757b601 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d7/FieldInstance.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d7/FieldInstance.php
@@ -41,7 +41,8 @@
  * In this example field instances of page content type are retrieved from the
  * source database.
  *
- * For additional configuration keys, refer to the parent classes:
+ * For additional configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d7/FieldInstancePerFormDisplay.php b/web/core/modules/field/src/Plugin/migrate/source/d7/FieldInstancePerFormDisplay.php
index 0ea30440a3545514a5a26e5d58c695368483cb09..e1f999718f8887a101cd988bbf63cbc172e76874 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d7/FieldInstancePerFormDisplay.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d7/FieldInstancePerFormDisplay.php
@@ -5,7 +5,8 @@
 /**
  * Drupal 7 field instance per form display source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\field\Plugin\migrate\source\d7\FieldInstance
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d7/FieldInstancePerViewMode.php b/web/core/modules/field/src/Plugin/migrate/source/d7/FieldInstancePerViewMode.php
index 9e38bb247d7aa3aaf007e500fb6acda3b2b0d733..44a0b42ec41b298e4f032b0464466cda6767904a 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d7/FieldInstancePerViewMode.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d7/FieldInstancePerViewMode.php
@@ -5,7 +5,8 @@
 /**
  * Drupal 7 field instance per view mode source class.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\field\Plugin\migrate\source\d7\FieldInstance
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d7/FieldLabelDescriptionTranslation.php b/web/core/modules/field/src/Plugin/migrate/source/d7/FieldLabelDescriptionTranslation.php
index 1ea718af50a2803b698e2db52f75a79dfa9024e3..bec150fe78e3b323a4e41886be806224bc487533 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d7/FieldLabelDescriptionTranslation.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d7/FieldLabelDescriptionTranslation.php
@@ -7,7 +7,8 @@
 /**
  * Drupal 7 i18n field label and description source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/field/src/Plugin/migrate/source/d7/FieldOptionTranslation.php b/web/core/modules/field/src/Plugin/migrate/source/d7/FieldOptionTranslation.php
index 5569541d2d43f5590500595a2bcf65526aa4a9ab..b7159b31d4fea107d3119905155415404324b95e 100644
--- a/web/core/modules/field/src/Plugin/migrate/source/d7/FieldOptionTranslation.php
+++ b/web/core/modules/field/src/Plugin/migrate/source/d7/FieldOptionTranslation.php
@@ -5,7 +5,8 @@
 /**
  * Drupal 7 i18n field option label source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/jsonapi/src/Controller/EntityResource.php b/web/core/modules/jsonapi/src/Controller/EntityResource.php
index 47bc9c056dedd941c3efba054de0750c39506c71..ca588d12f7a2e180d0a5d3bcdc0b651ebf15d759 100644
--- a/web/core/modules/jsonapi/src/Controller/EntityResource.php
+++ b/web/core/modules/jsonapi/src/Controller/EntityResource.php
@@ -369,7 +369,25 @@ public function patchIndividual(ResourceType $resource_type, EntityInterface $en
    *   The response.
    */
   public function deleteIndividual(EntityInterface $entity) {
-    $entity->delete();
+    // @todo Replace with entity handlers in: https://www.drupal.org/project/drupal/issues/3230434
+    if ($entity->getEntityTypeId() === 'user') {
+      $cancel_method = \Drupal::service('config.factory')->get('user.settings')->get('cancel_method');
+
+      // Allow other modules to act.
+
+      user_cancel([], $entity->id(), $cancel_method);
+      // Since user_cancel() is not invoked via Form API, batch processing
+      // needs to be invoked manually.
+      $batch =& batch_get();
+      // Mark this batch as non-progressive to bypass the progress bar and
+      // redirect.
+      $batch['progressive'] = FALSE;
+      batch_process();
+    }
+    else {
+      $entity->delete();
+    }
+
     return new ResourceResponse(NULL, 204);
   }
 
diff --git a/web/core/modules/jsonapi/tests/src/Functional/JsonApiFunctionalDateFieldTest.php b/web/core/modules/jsonapi/tests/src/Functional/JsonApiFunctionalDateFieldTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..21b2a7b9a39b6a526050ff94731388a589086660
--- /dev/null
+++ b/web/core/modules/jsonapi/tests/src/Functional/JsonApiFunctionalDateFieldTest.php
@@ -0,0 +1,182 @@
+<?php
+
+namespace Drupal\Tests\jsonapi\Functional;
+
+use Drupal\Component\Serialization\Json;
+use Drupal\datetime\Plugin\Field\FieldType\DateTimeItemInterface;
+use Drupal\field\Entity\FieldConfig;
+use Drupal\field\Entity\FieldStorageConfig;
+use Drupal\node\NodeInterface;
+
+/**
+ * JSON:API integration test for the "Date" field.
+ *
+ * @group jsonapi
+ */
+class JsonApiFunctionalDateFieldTest extends JsonApiFunctionalTestBase {
+
+  /**
+   * {@inheritdoc}
+   */
+  protected static $modules = [
+    'basic_auth',
+    'datetime',
+  ];
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function setUp(): void {
+    parent::setUp();
+
+    FieldStorageConfig::create([
+      'field_name' => 'field_datetime',
+      'entity_type' => 'node',
+      'type' => 'datetime',
+      'settings' => [
+        'datetime_type' => 'datetime',
+      ],
+      'cardinality' => 1,
+    ])->save();
+
+    FieldConfig::create([
+      'field_name' => 'field_datetime',
+      'label' => 'Date and time',
+      'entity_type' => 'node',
+      'bundle' => 'article',
+      'required' => FALSE,
+      'settings' => [],
+      'description' => '',
+    ])->save();
+  }
+
+  /**
+   * Tests the GET method.
+   */
+  public function testRead() {
+    /** @var \Drupal\Core\Datetime\DateFormatterInterface $date_formatter */
+    $date_formatter = $this->container->get('date.formatter');
+
+    $timestamp_1 = '5000000';
+    $timestamp_2 = '6000000';
+    $timestamp_3 = '7000000';
+    // Expected: node 1.
+    $timestamp_smaller_than_value = $timestamp_2;
+    // Expected: node 1 and node 2.
+    $timestamp_smaller_than_or_equal_value = $timestamp_2;
+    // Expected: node 3.
+    $timestamp_greater_than_value = $timestamp_2;
+    // Expected: node 2 and node 3.
+    $timestamp_greater_than_or_equal_value = $timestamp_2;
+
+    $node_1 = $this->createNode([
+      'type' => 'article',
+      'uuid' => 'es_test_1',
+      'status' => NodeInterface::PUBLISHED,
+      'field_datetime' => $date_formatter->format($timestamp_1, 'custom', DateTimeItemInterface::DATETIME_STORAGE_FORMAT),
+    ]);
+    $node_2 = $this->createNode([
+      'type' => 'article',
+      'uuid' => 'es_test_2',
+      'status' => NodeInterface::PUBLISHED,
+      'field_datetime' => $date_formatter->format($timestamp_2, 'custom', DateTimeItemInterface::DATETIME_STORAGE_FORMAT),
+    ]);
+    $node_3 = $this->createNode([
+      'type' => 'article',
+      'uuid' => 'es_test_3',
+      'status' => NodeInterface::PUBLISHED,
+      'field_datetime' => $date_formatter->format($timestamp_3, 'custom', DateTimeItemInterface::DATETIME_STORAGE_FORMAT),
+    ]);
+
+    // Checks whether the date is greater than the given timestamp.
+    $filter = [
+      'filter_datetime' => [
+        'condition' => [
+          'path' => 'field_datetime',
+          'operator' => '>',
+          'value' => date(DateTimeItemInterface::DATETIME_STORAGE_FORMAT, $timestamp_greater_than_value),
+        ],
+      ],
+    ];
+    $output = Json::decode($this->drupalGet('/jsonapi/node/article', [
+      'query' => ['filter' => $filter],
+    ]));
+    $this->assertSession()->statusCodeEquals(200);
+    $output_uuids = array_map(function ($result) {
+      return $result['id'];
+    }, $output['data']);
+    $this->assertCount(1, $output_uuids);
+    $this->assertSame([
+      $node_3->uuid(),
+    ], $output_uuids);
+
+    // Checks whether the date is greater than or equal to the given timestamp.
+    $filter = [
+      'filter_datetime' => [
+        'condition' => [
+          'path' => 'field_datetime',
+          'operator' => '>=',
+          'value' => date(DateTimeItemInterface::DATETIME_STORAGE_FORMAT, $timestamp_greater_than_or_equal_value),
+        ],
+      ],
+    ];
+    $output = Json::decode($this->drupalGet('/jsonapi/node/article', [
+      'query' => ['filter' => $filter],
+    ]));
+    $this->assertSession()->statusCodeEquals(200);
+    $output_uuids = array_map(function ($result) {
+      return $result['id'];
+    }, $output['data']);
+    $this->assertCount(2, $output_uuids);
+    $this->assertSame([
+      $node_2->uuid(),
+      $node_3->uuid(),
+    ], $output_uuids);
+
+    // Checks whether the date is less than the given timestamp.
+    $filter = [
+      'filter_datetime' => [
+        'condition' => [
+          'path' => 'field_datetime',
+          'operator' => '<',
+          'value' => date(DateTimeItemInterface::DATETIME_STORAGE_FORMAT, $timestamp_smaller_than_value),
+        ],
+      ],
+    ];
+    $output = Json::decode($this->drupalGet('/jsonapi/node/article', [
+      'query' => ['filter' => $filter],
+    ]));
+    $this->assertSession()->statusCodeEquals(200);
+    $output_uuids = array_map(function ($result) {
+      return $result['id'];
+    }, $output['data']);
+    $this->assertCount(1, $output_uuids);
+    $this->assertSame([
+      $node_1->uuid(),
+    ], $output_uuids);
+
+    // Checks whether the date is less than or equal to the given timestamp.
+    $filter = [
+      'filter_datetime' => [
+        'condition' => [
+          'path' => 'field_datetime',
+          'operator' => '<=',
+          'value' => date(DateTimeItemInterface::DATETIME_STORAGE_FORMAT, $timestamp_smaller_than_or_equal_value),
+        ],
+      ],
+    ];
+    $output = Json::decode($this->drupalGet('/jsonapi/node/article', [
+      'query' => ['filter' => $filter],
+    ]));
+    $this->assertSession()->statusCodeEquals(200);
+    $output_uuids = array_map(function ($result) {
+      return $result['id'];
+    }, $output['data']);
+    $this->assertCount(2, $output_uuids);
+    $this->assertSame([
+      $node_1->uuid(),
+      $node_2->uuid(),
+    ], $output_uuids);
+  }
+
+}
diff --git a/web/core/modules/jsonapi/tests/src/Functional/UserTest.php b/web/core/modules/jsonapi/tests/src/Functional/UserTest.php
index bc6b453b4317533e4f57b300d48531b18a944030..2184c3bcb4b8e0a46e99e5d75b4cc10d20e1c2ff 100644
--- a/web/core/modules/jsonapi/tests/src/Functional/UserTest.php
+++ b/web/core/modules/jsonapi/tests/src/Functional/UserTest.php
@@ -10,6 +10,7 @@
 use Drupal\field\Entity\FieldStorageConfig;
 use Drupal\node\Entity\Node;
 use Drupal\user\Entity\User;
+use Drupal\user\UserInterface;
 use GuzzleHttp\RequestOptions;
 
 /**
@@ -19,10 +20,12 @@
  */
 class UserTest extends ResourceTestBase {
 
+  const BATCH_TEST_NODE_COUNT = 15;
+
   /**
    * {@inheritdoc}
    */
-  protected static $modules = ['user', 'jsonapi_test_user'];
+  protected static $modules = ['user', 'jsonapi_test_user', 'node'];
 
   /**
    * {@inheritdoc}
@@ -119,6 +122,15 @@ protected function createAnotherEntity($key) {
     return $user;
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  public function testDeleteIndividual() {
+    $this->config('user.settings')->set('cancel_method', 'user_cancel_delete')->save(TRUE);
+
+    parent::testDeleteIndividual();
+  }
+
   /**
    * {@inheritdoc}
    */
@@ -597,6 +609,175 @@ public function testResaveAccountName() {
     $this->assertEquals($original_name, $updated_user->get('name')->value);
   }
 
+  /**
+   * Tests if JSON:API respects user.settings.cancel_method: user_cancel_block.
+   */
+  public function testDeleteRespectsUserCancelBlock() {
+    $cancel_method = 'user_cancel_block';
+    $this->config('jsonapi.settings')->set('read_only', FALSE)->save(TRUE);
+    $this->config('user.settings')->set('cancel_method', $cancel_method)->save(TRUE);
+
+    $account = $this->createAnotherEntity($cancel_method);
+    $node = $this->drupalCreateNode(['uid' => $account->id()]);
+
+    $this->sendDeleteRequestForUser($account, $cancel_method);
+
+    $user_storage = $this->container->get('entity_type.manager')
+      ->getStorage('user');
+    $user_storage->resetCache([$account->id()]);
+    $account = $user_storage->load($account->id());
+
+    $this->assertNotNull($account, 'User is not deleted after JSON:API DELETE operation with user.settings.cancel_method: ' . $cancel_method);
+    $this->assertTrue($account->isBlocked(), 'User is blocked after JSON:API DELETE operation with user.settings.cancel_method: ' . $cancel_method);
+
+    $node_storage = $this->container->get('entity_type.manager')->getStorage('node');
+    $node_storage->resetCache([$node->id()]);
+    $test_node = $node_storage->load($node->id());
+    $this->assertNotNull($test_node, 'Node of the user is not deleted.');
+    $this->assertTrue($test_node->isPublished(), 'Node of the user is published.');
+    $test_node = node_revision_load($node->getRevisionId());
+    $this->assertTrue($test_node->isPublished(), 'Node revision of the user is published.');
+  }
+
+  /**
+   * Tests if JSON:API respects user.settings.cancel_method: user_cancel_block_unpublish.
+   */
+  public function testDeleteRespectsUserCancelBlockUnpublish() {
+    $cancel_method = 'user_cancel_block_unpublish';
+    $this->config('jsonapi.settings')->set('read_only', FALSE)->save(TRUE);
+    $this->config('user.settings')->set('cancel_method', $cancel_method)->save(TRUE);
+
+    $account = $this->createAnotherEntity($cancel_method);
+    $node = $this->drupalCreateNode(['uid' => $account->id()]);
+
+    $this->sendDeleteRequestForUser($account, $cancel_method);
+
+    $user_storage = $this->container->get('entity_type.manager')
+      ->getStorage('user');
+    $user_storage->resetCache([$account->id()]);
+    $account = $user_storage->load($account->id());
+
+    $this->assertNotNull($account, 'User is not deleted after JSON:API DELETE operation with user.settings.cancel_method: ' . $cancel_method);
+    $this->assertTrue($account->isBlocked(), 'User is blocked after JSON:API DELETE operation with user.settings.cancel_method: ' . $cancel_method);
+
+    $node_storage = $this->container->get('entity_type.manager')->getStorage('node');
+    $node_storage->resetCache([$node->id()]);
+    $test_node = $node_storage->load($node->id());
+    $this->assertNotNull($test_node, 'Node of the user is not deleted.');
+    $this->assertFalse($test_node->isPublished(), 'Node of the user is no longer published.');
+    $test_node = node_revision_load($node->getRevisionId());
+    $this->assertFalse($test_node->isPublished(), 'Node revision of the user is no longer published.');
+  }
+
+  /**
+   * Tests if JSON:API respects user.settings.cancel_method: user_cancel_block_unpublish.
+   * @group jsonapi
+   */
+  public function testDeleteRespectsUserCancelBlockUnpublishAndProcessesBatches() {
+    $cancel_method = 'user_cancel_block_unpublish';
+    $this->config('jsonapi.settings')->set('read_only', FALSE)->save(TRUE);
+    $this->config('user.settings')->set('cancel_method', $cancel_method)->save(TRUE);
+
+    $account = $this->createAnotherEntity($cancel_method);
+
+    $nodeCount = self::BATCH_TEST_NODE_COUNT;
+    $node_ids = [];
+    $nodes = [];
+    while ($nodeCount-- > 0) {
+      $node = $this->drupalCreateNode(['uid' => $account->id()]);
+      $nodes[] = $node;
+      $node_ids[] = $node->id();
+    }
+
+    $this->sendDeleteRequestForUser($account, $cancel_method);
+
+    $user_storage = $this->container->get('entity_type.manager')
+      ->getStorage('user');
+    $user_storage->resetCache([$account->id()]);
+    $account = $user_storage->load($account->id());
+
+    $this->assertNotNull($account, 'User is not deleted after JSON:API DELETE operation with user.settings.cancel_method: ' . $cancel_method);
+    $this->assertTrue($account->isBlocked(), 'User is blocked after JSON:API DELETE operation with user.settings.cancel_method: ' . $cancel_method);
+
+    $node_storage = $this->container->get('entity_type.manager')->getStorage('node');
+    $node_storage->resetCache($node_ids);
+
+    $test_nodes = $node_storage->loadMultiple($node_ids);
+
+    $this->assertCount(self::BATCH_TEST_NODE_COUNT, $test_nodes, 'Nodes of the user are not deleted.');
+
+    foreach ($test_nodes as $test_node) {
+      $this->assertFalse($test_node->isPublished(), 'Node of the user is no longer published.');
+    }
+
+    foreach ($nodes as $node) {
+      $test_node = node_revision_load($node->getRevisionId());
+      $this->assertFalse($test_node->isPublished(), 'Node revision of the user is no longer published.');
+    }
+  }
+
+  /**
+   * Tests if JSON:API respects user.settings.cancel_method: user_cancel_reassign.
+   */
+  public function testDeleteRespectsUserCancelReassign() {
+    $cancel_method = 'user_cancel_reassign';
+    $this->config('jsonapi.settings')->set('read_only', FALSE)->save(TRUE);
+    $this->config('user.settings')->set('cancel_method', $cancel_method)->save(TRUE);
+
+    $account = $this->createAnotherEntity($cancel_method);
+    $node = $this->drupalCreateNode(['uid' => $account->id()]);
+
+    $this->sendDeleteRequestForUser($account, $cancel_method);
+
+    $user_storage = $this->container->get('entity_type.manager')
+      ->getStorage('user');
+    $user_storage->resetCache([$account->id()]);
+    $account = $user_storage->load($account->id());
+
+    $this->assertNull($account, 'User is deleted after JSON:API DELETE operation with user.settings.cancel_method: ' . $cancel_method);
+
+    $node_storage = $this->container->get('entity_type.manager')->getStorage('node');
+    $node_storage->resetCache([$node->id()]);
+    $test_node = $node_storage->load($node->id());
+    $this->assertNotNull($test_node, 'Node of the user is not deleted.');
+    $this->assertTrue($test_node->isPublished(), 'Node of the user is still published.');
+    $this->assertEquals(0, $test_node->getOwnerId(), 'Node of the user has been attributed to anonymous user.');
+    $test_node = node_revision_load($node->getRevisionId());
+    $this->assertTrue($test_node->isPublished(), 'Node revision of the user is still published.');
+    $this->assertEquals(0, $test_node->getRevisionUser()->id(), 'Node revision of the user has been attributed to anonymous user.');
+  }
+
+  /**
+   * Tests if JSON:API respects user.settings.cancel_method: user_cancel_delete.
+   */
+  public function testDeleteRespectsUserCancelDelete() {
+    $cancel_method = 'user_cancel_delete';
+    $this->config('jsonapi.settings')->set('read_only', FALSE)->save(TRUE);
+    $this->config('user.settings')->set('cancel_method', $cancel_method)->save(TRUE);
+
+    $account = $this->createAnotherEntity($cancel_method);
+    $node = $this->drupalCreateNode(['uid' => $account->id()]);
+
+    $url = Url::fromRoute(sprintf('jsonapi.%s.individual', static::$resourceTypeName), ['entity' => $account->uuid()]);
+    $request_options = [];
+    $request_options[RequestOptions::HEADERS]['Accept'] = 'application/vnd.api+json';
+    $request_options = NestedArray::mergeDeep($request_options, $this->getAuthenticationRequestOptions());
+    $this->setUpAuthorization('DELETE');
+    $response = $this->request('DELETE', $url, $request_options);
+    $this->assertResourceResponse(204, NULL, $response);
+
+    $node_storage = $this->container->get('entity_type.manager')->getStorage('node');
+    $user_storage = $this->container->get('entity_type.manager')->getStorage('user');
+
+    $user_storage->resetCache([$account->id()]);
+    $account = $user_storage->load($account->id());
+    $this->assertNull($account, 'User is deleted after JSON:API DELETE operation with user.settings.cancel_method: ' . $cancel_method);
+
+    $node_storage->resetCache([$node->id()]);
+    $test_node = $node_storage->load($node->id());
+    $this->assertNull($test_node, 'Node of the user is deleted.');
+  }
+
   /**
    * {@inheritdoc}
    */
@@ -620,4 +801,18 @@ protected function makeNormalizationInvalid(array $document, $entity_key) {
     return parent::makeNormalizationInvalid($document, $entity_key);
   }
 
+  /**
+   * @param \Drupal\user\UserInterface $account
+   * @param string $cancel_method
+   */
+  private function sendDeleteRequestForUser(UserInterface $account, string $cancel_method) {
+    $url = Url::fromRoute(sprintf('jsonapi.%s.individual', static::$resourceTypeName), ['entity' => $account->uuid()]);
+    $request_options = [];
+    $request_options[RequestOptions::HEADERS]['Accept'] = 'application/vnd.api+json';
+    $request_options = NestedArray::mergeDeep($request_options, $this->getAuthenticationRequestOptions());
+    $this->setUpAuthorization('DELETE');
+    $response = $this->request('DELETE', $url, $request_options);
+    $this->assertResourceResponse(204, NULL, $response);
+  }
+
 }
diff --git a/web/core/modules/language/src/Plugin/migrate/source/Language.php b/web/core/modules/language/src/Plugin/migrate/source/Language.php
index 9576cf7a79db939aff90e9fc5868cbd77d20e32b..870f9790e6591efcdd54bb1a434c558041130a20 100644
--- a/web/core/modules/language/src/Plugin/migrate/source/Language.php
+++ b/web/core/modules/language/src/Plugin/migrate/source/Language.php
@@ -6,7 +6,7 @@
 use Drupal\migrate_drupal\Plugin\migrate\source\DrupalSqlBase;
 
 /**
- * Drupal language source from database.
+ * Drupal 6/7 language source from database.
  *
  * Available configuration keys:
  * - fetch_all: (optional) If not empty, all source languages are retrieved and
@@ -28,7 +28,8 @@
  * Given that fetch_all and domain_negotiation are specified, each row also
  * contains all languages and the domain negotiation status, if enabled.
  *
- * For additional configuration keys, refer to the parent classes:
+ * For additional configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/language/src/Plugin/migrate/source/d6/LanguageContentSettings.php b/web/core/modules/language/src/Plugin/migrate/source/d6/LanguageContentSettings.php
index 9fbe85c831f3117e9f86cef7c8fe22d081b32770..d2f4c8b342b1cfc2b6064a68d57dd2ca2f4767af 100644
--- a/web/core/modules/language/src/Plugin/migrate/source/d6/LanguageContentSettings.php
+++ b/web/core/modules/language/src/Plugin/migrate/source/d6/LanguageContentSettings.php
@@ -8,7 +8,8 @@
 /**
  * Drupal 6 i18n node settings from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBas
  *
diff --git a/web/core/modules/language/src/Plugin/migrate/source/d6/LanguageContentSettingsTaxonomyVocabulary.php b/web/core/modules/language/src/Plugin/migrate/source/d6/LanguageContentSettingsTaxonomyVocabulary.php
index 246e635ddf2b16fcde459fb35f87def7567dce91..4eddde832ec42d497eef48f76360151a26b94024 100644
--- a/web/core/modules/language/src/Plugin/migrate/source/d6/LanguageContentSettingsTaxonomyVocabulary.php
+++ b/web/core/modules/language/src/Plugin/migrate/source/d6/LanguageContentSettingsTaxonomyVocabulary.php
@@ -8,7 +8,8 @@
 /**
  * Drupal 6 i18n vocabularies source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBas
  *
diff --git a/web/core/modules/language/src/Plugin/migrate/source/d7/LanguageContentSettings.php b/web/core/modules/language/src/Plugin/migrate/source/d7/LanguageContentSettings.php
index 6ce9283c5186980e4a53248cf341be7c36897e9b..ae0b11409d3e075a597162d522ac979993e7148f 100644
--- a/web/core/modules/language/src/Plugin/migrate/source/d7/LanguageContentSettings.php
+++ b/web/core/modules/language/src/Plugin/migrate/source/d7/LanguageContentSettings.php
@@ -8,7 +8,8 @@
 /**
  * Drupal 7 i18n node settings from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBas
  *
diff --git a/web/core/modules/language/src/Plugin/migrate/source/d7/LanguageContentSettingsTaxonomyVocabulary.php b/web/core/modules/language/src/Plugin/migrate/source/d7/LanguageContentSettingsTaxonomyVocabulary.php
index 9017a9e461d5b5d4ca253b67894ef7ad8359fa39..89622ab5b0100f7078a7e2a6db995ad6f75060b5 100644
--- a/web/core/modules/language/src/Plugin/migrate/source/d7/LanguageContentSettingsTaxonomyVocabulary.php
+++ b/web/core/modules/language/src/Plugin/migrate/source/d7/LanguageContentSettingsTaxonomyVocabulary.php
@@ -7,7 +7,8 @@
 /**
  * Drupal 7 i18n vocabularies source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBas
  *
diff --git a/web/core/modules/language/tests/src/Kernel/Migrate/d7/MigrateLanguageContentCommentSettingsNoEntityTranslationTest.php b/web/core/modules/language/tests/src/Kernel/Migrate/d7/MigrateLanguageContentCommentSettingsNoEntityTranslationTest.php
index 287a98e2fdfcb6fa4beae989a798dadd296a1973..29cb946832e8da9bce759dc4d6a85996999498c5 100644
--- a/web/core/modules/language/tests/src/Kernel/Migrate/d7/MigrateLanguageContentCommentSettingsNoEntityTranslationTest.php
+++ b/web/core/modules/language/tests/src/Kernel/Migrate/d7/MigrateLanguageContentCommentSettingsNoEntityTranslationTest.php
@@ -31,6 +31,7 @@ class MigrateLanguageContentCommentSettingsNoEntityTranslationTest extends Migra
    */
   protected function setUp(): void {
     parent::setUp();
+    $this->startCollectingMessages();
     $this->migrateCommentTypes();
     $this->executeMigrations([
       'language',
@@ -57,6 +58,9 @@ public static function migrateDumpAlter(KernelTestBase $test) {
    * Tests migration of content language settings.
    */
   public function testLanguageCommentSettings() {
+    // Confirm there is no message about a missing bundle.
+    $this->assertEmpty($this->migrateMessages, $this->migrateMessages['error'][0] ?? '');
+
     // Article and Blog content type have multilingual settings of 'Enabled,
     // with Translation'. Assert that comments are translatable and the default
     // language is 'current_interface'.
@@ -88,6 +92,15 @@ public function testLanguageCommentSettings() {
     $this->assertTrue($config->isLanguageAlterable());
     $this->assertSame($third_party_settings, $config->get('third_party_settings'));
 
+    // The content type with a long name has multilingual settings of 'Enabled'.
+    $config = ContentLanguageSettings::loadByEntityTypeBundle('comment', 'comment_node_a_thirty_two_char');
+    $this->assertFalse($config->isNew());
+    $this->assertSame('comment', $config->getTargetEntityTypeId());
+    $this->assertSame('comment_node_a_thirty_two_char', $config->getTargetBundle());
+    $this->assertSame('current_interface', $config->getDefaultLangcode());
+    $this->assertTrue($config->isLanguageAlterable());
+    $this->assertSame($third_party_settings, $config->get('third_party_settings'));
+
     // Test content type has multilingual settings of 'Enabled, with field
     // translation'.
     $config = ContentLanguageSettings::loadByEntityTypeBundle('comment', 'comment_node_test_content_type');
diff --git a/web/core/modules/language/tests/src/Kernel/Migrate/d7/MigrateLanguageContentCommentSettingsTest.php b/web/core/modules/language/tests/src/Kernel/Migrate/d7/MigrateLanguageContentCommentSettingsTest.php
index 4032c9dd05ba95684d21c1c8f9570be7574ff905..db157140ced568544005c68e6b4c1d8ff10c4fb0 100644
--- a/web/core/modules/language/tests/src/Kernel/Migrate/d7/MigrateLanguageContentCommentSettingsTest.php
+++ b/web/core/modules/language/tests/src/Kernel/Migrate/d7/MigrateLanguageContentCommentSettingsTest.php
@@ -28,6 +28,7 @@ class MigrateLanguageContentCommentSettingsTest extends MigrateDrupal7TestBase {
    */
   protected function setUp(): void {
     parent::setUp();
+    $this->startCollectingMessages();
     $this->migrateCommentTypes();
     $this->executeMigrations([
       'language',
@@ -39,6 +40,9 @@ protected function setUp(): void {
    * Tests migration of content language settings.
    */
   public function testLanguageCommentSettings() {
+    // Confirm there is no message about a missing bundle.
+    $this->assertEmpty($this->migrateMessages, $this->migrateMessages['error'][0] ?? '');
+
     // Article and Blog content type have multilingual settings of 'Enabled,
     // with Translation'. Assert that comments are translatable and the default
     // language is 'current_interface'.
@@ -70,6 +74,15 @@ public function testLanguageCommentSettings() {
     $this->assertTrue($config->isLanguageAlterable());
     $this->assertSame($third_party_settings, $config->get('third_party_settings'));
 
+    // The content type with a long name has multilingual settings of 'Enabled'.
+    $config = ContentLanguageSettings::loadByEntityTypeBundle('comment', 'comment_node_a_thirty_two_char');
+    $this->assertFalse($config->isNew());
+    $this->assertSame('comment', $config->getTargetEntityTypeId());
+    $this->assertSame('comment_node_a_thirty_two_char', $config->getTargetBundle());
+    $this->assertSame('current_interface', $config->getDefaultLangcode());
+    $this->assertTrue($config->isLanguageAlterable());
+    $this->assertSame($third_party_settings, $config->get('third_party_settings'));
+
     // Test content type has multilingual settings of 'Enabled, with field
     // translation'.
     $config = ContentLanguageSettings::loadByEntityTypeBundle('comment', 'comment_node_test_content_type');
diff --git a/web/core/modules/layout_builder/src/EventSubscriber/BlockComponentRenderArray.php b/web/core/modules/layout_builder/src/EventSubscriber/BlockComponentRenderArray.php
index d9218b538776654f766fdc1bbba2c1e78012e5c6..8ed8137f253f5ad1dfb0d9f3e029ecf3dbb89995 100644
--- a/web/core/modules/layout_builder/src/EventSubscriber/BlockComponentRenderArray.php
+++ b/web/core/modules/layout_builder/src/EventSubscriber/BlockComponentRenderArray.php
@@ -12,6 +12,7 @@
 use Drupal\Core\StringTranslation\StringTranslationTrait;
 use Drupal\layout_builder\Access\LayoutPreviewAccessAllowed;
 use Drupal\layout_builder\Event\SectionComponentBuildRenderArrayEvent;
+use Drupal\layout_builder\Plugin\Block\InlineBlock;
 use Drupal\layout_builder\LayoutBuilderEvents;
 use Drupal\views\Plugin\Block\ViewsBlock;
 use Symfony\Component\EventDispatcher\EventSubscriberInterface;
@@ -124,9 +125,31 @@ public function onBuildRender(SectionComponentBuildRenderArrayEvent $event) {
         '#base_plugin_id' => $block->getBaseId(),
         '#derivative_plugin_id' => $block->getDerivativeId(),
         '#weight' => $event->getComponent()->getWeight(),
-        'content' => $content,
       ];
 
+      // Place the $content returned by the block plugin into a 'content' child
+      // element, as a way to allow the plugin to have complete control of its
+      // properties and rendering (for instance, its own #theme) without
+      // conflicting with the properties used above, or alternate ones used by
+      // alternate block rendering approaches in contributed modules. However,
+      // the use of a child element is an implementation detail of this
+      // particular block rendering approach. Semantically, the content returned
+      // by the block plugin, and in particular, attributes and contextual links
+      // are information that belong to the entire block. Therefore, we must
+      // move these properties from $content and merge them into the top-level
+      // element.
+      if (isset($content['#attributes'])) {
+        $build['#attributes'] = $content['#attributes'];
+        unset($content['#attributes']);
+      }
+      // Hide contextual links for inline blocks until the UX issues surrounding
+      // editing them directly are resolved.
+      // @see https://www.drupal.org/project/drupal/issues/3075308
+      if (!$block instanceof InlineBlock && !empty($content['#contextual_links'])) {
+        $build['#contextual_links'] = $content['#contextual_links'];
+      }
+      $build['content'] = $content;
+
       if ($event->inPreview()) {
         if ($block instanceof PreviewFallbackInterface) {
           $preview_fallback_string = $block->getPreviewFallbackString();
diff --git a/web/core/modules/layout_builder/tests/modules/layout_builder_test/layout_builder_test.links.contextual.yml b/web/core/modules/layout_builder/tests/modules/layout_builder_test/layout_builder_test.links.contextual.yml
new file mode 100644
index 0000000000000000000000000000000000000000..62a4a509dd3c334855e06abe701221af98a00c81
--- /dev/null
+++ b/web/core/modules/layout_builder/tests/modules/layout_builder_test/layout_builder_test.links.contextual.yml
@@ -0,0 +1,4 @@
+layout_builder_test:
+  title: 'Test contextual link'
+  route_name: '<front>'
+  group: 'layout_builder_test'
diff --git a/web/core/modules/layout_builder/tests/modules/layout_builder_test/src/Plugin/Block/TestAttributesBlock.php b/web/core/modules/layout_builder/tests/modules/layout_builder_test/src/Plugin/Block/TestAttributesBlock.php
new file mode 100644
index 0000000000000000000000000000000000000000..877fc3bfe7fa51a421893a05bc37a5246dad02ce
--- /dev/null
+++ b/web/core/modules/layout_builder/tests/modules/layout_builder_test/src/Plugin/Block/TestAttributesBlock.php
@@ -0,0 +1,43 @@
+<?php
+
+namespace Drupal\layout_builder_test\Plugin\Block;
+
+use Drupal\Core\Block\BlockBase;
+use Drupal\Core\Form\FormStateInterface;
+
+/**
+ * Provides a 'TestAttributes' block.
+ *
+ * @Block(
+ *   id = "layout_builder_test_test_attributes",
+ *   admin_label = @Translation("Test Attributes"),
+ *   category = @Translation("Test")
+ * )
+ */
+class TestAttributesBlock extends BlockBase {
+
+  /**
+   * {@inheritdoc}
+   */
+  public function blockForm($form, FormStateInterface $form_state) {
+    return $form;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function build() {
+    $build = [
+      '#attributes' => [
+        'class' => ['attribute-test-class'],
+        'custom-attribute' => 'test',
+      ],
+      '#markup' => $this->t('Example block providing its own attributes.'),
+      '#contextual_links' => [
+        'layout_builder_test' => ['route_parameters' => []],
+      ],
+    ];
+    return $build;
+  }
+
+}
diff --git a/web/core/modules/layout_builder/tests/src/Functional/LayoutBuilderTest.php b/web/core/modules/layout_builder/tests/src/Functional/LayoutBuilderTest.php
index ba6f5bc41c48e369a36c5ef39f5a6511b9a0eece..8f121c75b4738bc525de07e3267ea1c4d943739b 100644
--- a/web/core/modules/layout_builder/tests/src/Functional/LayoutBuilderTest.php
+++ b/web/core/modules/layout_builder/tests/src/Functional/LayoutBuilderTest.php
@@ -25,6 +25,7 @@ class LayoutBuilderTest extends BrowserTestBase {
     'layout_test',
     'block',
     'block_test',
+    'contextual',
     'node',
     'layout_builder_test',
   ];
@@ -639,6 +640,37 @@ public function testPluginDependencies() {
     $assert_session->elementNotExists('css', '.block.menu--my-menu');
   }
 
+  /**
+   * Tests that block plugins can define custom attributes and contextual links.
+   */
+  public function testPluginsProvidingCustomAttributesAndContextualLinks() {
+    $assert_session = $this->assertSession();
+    $page = $this->getSession()->getPage();
+
+    $this->drupalLogin($this->drupalCreateUser([
+      'access contextual links',
+      'configure any layout',
+      'administer node display',
+    ]));
+
+    $this->drupalGet('admin/structure/types/manage/bundle_with_section_field/display/default');
+    $this->submitForm(['layout[enabled]' => TRUE], 'Save');
+    $page->clickLink('Manage layout');
+    $page->clickLink('Add section');
+    $page->clickLink('Layout Builder Test Plugin');
+    $page->pressButton('Add section');
+    $page->clickLink('Add block');
+    $page->clickLink('Test Attributes');
+    $page->pressButton('Add block');
+    $page->pressButton('Save layout');
+
+    $this->drupalGet('node/1');
+
+    $assert_session->elementExists('css', '.attribute-test-class');
+    $assert_session->elementExists('css', '[custom-attribute=test]');
+    $assert_session->elementExists('css', 'div[data-contextual-id*="layout_builder_test"]');
+  }
+
   /**
    * Tests the interaction between full and default view modes.
    *
diff --git a/web/core/modules/link/src/Plugin/migrate/process/FieldLink.php b/web/core/modules/link/src/Plugin/migrate/process/FieldLink.php
index 232ed903f9b582194b067e4c91979c7edaad8965..b118f449b1a80c51d4301b5add42241b4f1ab0ec 100644
--- a/web/core/modules/link/src/Plugin/migrate/process/FieldLink.php
+++ b/web/core/modules/link/src/Plugin/migrate/process/FieldLink.php
@@ -59,6 +59,13 @@ public function __construct(array $configuration, $plugin_id, $plugin_definition
    * @see \Drupal\link\Plugin\Field\FieldWidget\LinkWidget::getUserEnteredStringAsUri()
    */
   protected function canonicalizeUri($uri) {
+    // If the path starts with 2 slashes then it is always considered an
+    // external URL without an explicit protocol part.
+    // @todo Remove this when https://www.drupal.org/node/2744729 lands.
+    if (strpos($uri, '//') === 0) {
+      return $this->configuration['uri_scheme'] . ltrim($uri, '/');
+    }
+
     // If we already have a scheme, we're fine.
     if (parse_url($uri, PHP_URL_SCHEME)) {
       return $uri;
diff --git a/web/core/modules/link/tests/src/Unit/Plugin/migrate/process/FieldLinkTest.php b/web/core/modules/link/tests/src/Unit/Plugin/migrate/process/FieldLinkTest.php
index d6721e51f2e5f952493f9618244a72e62fa5517d..1142ec0c2005193b90ea6ecb852b2cbda6e7dff6 100644
--- a/web/core/modules/link/tests/src/Unit/Plugin/migrate/process/FieldLinkTest.php
+++ b/web/core/modules/link/tests/src/Unit/Plugin/migrate/process/FieldLinkTest.php
@@ -66,6 +66,10 @@ public function canonicalizeUriDataProvider() {
         'https://yahoo.com',
         ['uri_scheme' => 'https://'],
       ],
+      'Absolute URL without explicit protocol (protocol-relative)' => [
+        '//example.com',
+        'http://example.com',
+      ],
       'Absolute URL with non-standard characters' => [
         'http://www.ßÀÑÐ¥ƒå¢ë.com',
         'http://www.ßÀÑÐ¥ƒå¢ë.com',
diff --git a/web/core/modules/media/media.install b/web/core/modules/media/media.install
index f9fddb2511c9d5754159dd7a34029156876f1c93..a2c5eed95d1e758b6b1b9a424f7820184664bf77 100644
--- a/web/core/modules/media/media.install
+++ b/web/core/modules/media/media.install
@@ -120,22 +120,37 @@ function media_requirements($phase) {
       }
     }
 
-    // When a new media type with an image source is created we're configuring
-    // the default entity view display using the 'large' image style.
-    // Unfortunately, if a site builder has deleted the 'large' image style,
-    // we need some other image style to use, but at this point, we can't
-    // really know the site builder's intentions. So rather than do something
-    // surprising, we're leaving the embedded media without an image style and
-    // adding a warning that the site builder might want to add an image style.
-    // @see Drupal\media\Plugin\media\Source\Image::prepareViewDisplay
     $module_handler = \Drupal::service('module_handler');
     foreach (MediaType::loadMultiple() as $type) {
       // Load the default display.
       $display = \Drupal::service('entity_display.repository')
         ->getViewDisplay('media', $type->id());
 
+      // Check for missing source field definition.
       $source_field_definition = $type->getSource()->getSourceFieldDefinition($type);
-      if (empty($source_field_definition) || !is_a($source_field_definition->getItemDefinition()->getClass(), ImageItem::class, TRUE)) {
+      if (empty($source_field_definition)) {
+        $requirements['media_missing_source_field_' . $type->id()] = [
+          'title' => t('Media'),
+          'description' => t('The source field definition for the %type media type is missing.',
+            [
+              '%type' => $type->label(),
+            ]
+          ),
+          'severity' => REQUIREMENT_ERROR,
+        ];
+        continue;
+      }
+
+      // When a new media type with an image source is created we're
+      // configuring the default entity view display using the 'large' image
+      // style. Unfortunately, if a site builder has deleted the 'large' image
+      // style, we need some other image style to use, but at this point, we
+      // can't really know the site builder's intentions. So rather than do
+      // something surprising, we're leaving the embedded media without an
+      // image style and adding a warning that the site builder might want to
+      // add an image style.
+      // @see Drupal\media\Plugin\media\Source\Image::prepareViewDisplay
+      if (!is_a($source_field_definition->getItemDefinition()->getClass(), ImageItem::class, TRUE)) {
         continue;
       }
 
diff --git a/web/core/modules/media/src/Plugin/Field/FieldWidget/OEmbedWidget.php b/web/core/modules/media/src/Plugin/Field/FieldWidget/OEmbedWidget.php
index 1252265e45b89261f7bff3f5e0dfc8a16b49576a..9ca270c6a17b9e2b593a1a4bb9d4666ed446fb65 100644
--- a/web/core/modules/media/src/Plugin/Field/FieldWidget/OEmbedWidget.php
+++ b/web/core/modules/media/src/Plugin/Field/FieldWidget/OEmbedWidget.php
@@ -36,7 +36,7 @@ public function formElement(FieldItemListInterface $items, $delta, array $elemen
     $source = $items->getEntity()->getSource();
     $message = $this->t('You can link to media from the following services: @providers', ['@providers' => implode(', ', $source->getProviders())]);
 
-    if (!empty($element['#value']['#description'])) {
+    if (!empty($element['value']['#description'])) {
       $element['value']['#description'] = [
         '#theme' => 'item_list',
         '#items' => [$element['value']['#description'], $message],
diff --git a/web/core/modules/media/tests/src/Functional/FieldWidget/OEmbedFieldWidgetTest.php b/web/core/modules/media/tests/src/Functional/FieldWidget/OEmbedFieldWidgetTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..5d1caf11ae77667cdc728910d894ba5ec1597714
--- /dev/null
+++ b/web/core/modules/media/tests/src/Functional/FieldWidget/OEmbedFieldWidgetTest.php
@@ -0,0 +1,43 @@
+<?php
+
+namespace Drupal\Tests\media\Functional\FieldWidget;
+
+use Drupal\field\Entity\FieldConfig;
+use Drupal\Tests\media\Functional\MediaFunctionalTestBase;
+
+/**
+ * @covers \Drupal\media\Plugin\Field\FieldWidget\OEmbedWidget
+ *
+ * @group media
+ */
+class OEmbedFieldWidgetTest extends MediaFunctionalTestBase {
+
+  /**
+   * {@inheritdoc}
+   */
+  protected $defaultTheme = 'stark';
+
+  /**
+   * Tests that the oEmbed field widget shows the configured help text.
+   */
+  public function testFieldWidgetHelpText() {
+    $account = $this->drupalCreateUser(['create media']);
+    $this->drupalLogin($account);
+
+    $media_type = $this->createMediaType('oembed:video');
+    $source_field = $media_type->getSource()
+      ->getSourceFieldDefinition($media_type)
+      ->getName();
+
+    /** @var \Drupal\field\Entity\FieldConfig $field */
+    $field = FieldConfig::loadByName('media', $media_type->id(), $source_field);
+    $field->setDescription('This is help text for oEmbed field.')
+      ->save();
+
+    $this->drupalGet('media/add/' . $media_type->id());
+    $assert_session = $this->assertSession();
+    $assert_session->pageTextContains('This is help text for oEmbed field.');
+    $assert_session->pageTextContains('You can link to media from the following services: YouTube, Vimeo');
+  }
+
+}
diff --git a/web/core/modules/media/tests/src/Functional/MediaRequirementsTest.php b/web/core/modules/media/tests/src/Functional/MediaRequirementsTest.php
index e8ff550cbcb2ccde00f33b5e0b68477c955d0779..db12de615073f0cf1d1960c3822bfaf8b48a23cb 100644
--- a/web/core/modules/media/tests/src/Functional/MediaRequirementsTest.php
+++ b/web/core/modules/media/tests/src/Functional/MediaRequirementsTest.php
@@ -26,10 +26,13 @@ public function testMissingSourceFieldDefinition() {
     $field_storage_definition = $field_definition->getFieldStorageDefinition();
     $field_definition->delete();
     $field_storage_definition->delete();
+    $valid_media_type = $this->createMediaType('test');
 
     $this->drupalLogin($this->rootUser);
     $this->drupalGet('/admin/reports/status');
     $this->assertSession()->statusCodeEquals(200);
+    $this->assertSession()->pageTextContains("The source field definition for the {$media_type->label()} media type is missing.");
+    $this->assertSession()->pageTextNotContains("The source field definition for the {$valid_media_type->label()} media type is missing.");
   }
 
 }
diff --git a/web/core/modules/media_library/tests/src/FunctionalJavascript/CKEditorIntegrationTest.php b/web/core/modules/media_library/tests/src/FunctionalJavascript/CKEditorIntegrationTest.php
index ca95070bc775dee456d166fb0a8aec5890534052..40fe5f7f537c4bdd40518e94a7b86dad982301d8 100644
--- a/web/core/modules/media_library/tests/src/FunctionalJavascript/CKEditorIntegrationTest.php
+++ b/web/core/modules/media_library/tests/src/FunctionalJavascript/CKEditorIntegrationTest.php
@@ -236,8 +236,6 @@ public function testButton() {
     $this->assignNameToCkeditorIframe();
     $this->getSession()->switchToIFrame('ckeditor');
     $this->assertNotEmpty($assert_session->waitForElementVisible('css', '.cke_widget_drupalmedia drupal-media .media'));
-    // @todo Inserting media embed should enable undo.
-    // @see https://www.drupal.org/project/drupal/issues/3073294
     $this->pressEditorButton('source');
     $value = $assert_session->elementExists('css', 'textarea.cke_source')->getValue();
     $dom = Html::load($value);
diff --git a/web/core/modules/menu_link_content/src/Plugin/migrate/source/MenuLink.php b/web/core/modules/menu_link_content/src/Plugin/migrate/source/MenuLink.php
index 1b186a6e04a18749d3ccb7e09194364c5e485c44..ff413abf8a60d339feee76f32cc67b253b4d1cd7 100644
--- a/web/core/modules/menu_link_content/src/Plugin/migrate/source/MenuLink.php
+++ b/web/core/modules/menu_link_content/src/Plugin/migrate/source/MenuLink.php
@@ -7,7 +7,7 @@
 use Drupal\migrate\Row;
 
 /**
- * Drupal menu link source from database.
+ * Drupal 6/7 menu link source from database.
  *
  * @MigrateSource(
  *   id = "menu_link",
diff --git a/web/core/modules/menu_link_content/src/Plugin/migrate/source/d6/MenuLinkTranslation.php b/web/core/modules/menu_link_content/src/Plugin/migrate/source/d6/MenuLinkTranslation.php
index 54b20687342bdbec786c73dc7d9f4bdf685adc90..863e5bee494ad35d7d455a17d9df7caf05484a2e 100644
--- a/web/core/modules/menu_link_content/src/Plugin/migrate/source/d6/MenuLinkTranslation.php
+++ b/web/core/modules/menu_link_content/src/Plugin/migrate/source/d6/MenuLinkTranslation.php
@@ -7,7 +7,7 @@
 use Drupal\menu_link_content\Plugin\migrate\source\MenuLink;
 
 /**
- * Gets Menu link translations from source database.
+ * Drupal 6 i18n menu link translations source from database.
  *
  * @MigrateSource(
  *   id = "d6_menu_link_translation",
diff --git a/web/core/modules/menu_link_content/src/Plugin/migrate/source/d7/MenuLinkLocalized.php b/web/core/modules/menu_link_content/src/Plugin/migrate/source/d7/MenuLinkLocalized.php
index b683186cea1b9cd5a4c8f25e3df13cce355f03eb..0b9e5ce30f1c27349a0a8858b3ed4089378a885e 100644
--- a/web/core/modules/menu_link_content/src/Plugin/migrate/source/d7/MenuLinkLocalized.php
+++ b/web/core/modules/menu_link_content/src/Plugin/migrate/source/d7/MenuLinkLocalized.php
@@ -6,7 +6,7 @@
 use Drupal\migrate\Row;
 
 /**
- * Gets localized menu link translations from source database.
+ * Drupal 7 localized menu link translations source from database.
  *
  * @MigrateSource(
  *   id = "d7_menu_link_localized",
diff --git a/web/core/modules/menu_link_content/src/Plugin/migrate/source/d7/MenuLinkTranslation.php b/web/core/modules/menu_link_content/src/Plugin/migrate/source/d7/MenuLinkTranslation.php
index b55a1a218f57ed6e7f08fa12527ee880be38c455..267e896b9bcad6637ea9f42186eb117b1233a2b9 100644
--- a/web/core/modules/menu_link_content/src/Plugin/migrate/source/d7/MenuLinkTranslation.php
+++ b/web/core/modules/menu_link_content/src/Plugin/migrate/source/d7/MenuLinkTranslation.php
@@ -7,7 +7,7 @@
 use Drupal\menu_link_content\Plugin\migrate\source\MenuLink;
 
 /**
- * Gets Menu link translations from source database.
+ * Drupal 7 i18n menu link translations source from database.
  *
  * @MigrateSource(
  *   id = "d7_menu_link_translation",
diff --git a/web/core/modules/menu_ui/src/MenuListBuilder.php b/web/core/modules/menu_ui/src/MenuListBuilder.php
index c6efe07222775d9d52e283316f12cc23aa1d3ab5..59af3090bd47d0d914496301367e6215e48dc7e6 100644
--- a/web/core/modules/menu_ui/src/MenuListBuilder.php
+++ b/web/core/modules/menu_ui/src/MenuListBuilder.php
@@ -4,6 +4,7 @@
 
 use Drupal\Core\Config\Entity\ConfigEntityListBuilder;
 use Drupal\Core\Entity\EntityInterface;
+use Drupal\Core\Url;
 
 /**
  * Defines a class to build a listing of menu entities.
@@ -49,6 +50,9 @@ public function getDefaultOperations(EntityInterface $entity) {
         'title' => t('Add link'),
         'weight' => 20,
         'url' => $entity->toUrl('add-link-form'),
+        'query' => [
+          'destination' => $entity->toUrl('edit-form')->toString(),
+        ],
       ];
     }
     if (isset($operations['delete'])) {
@@ -57,6 +61,16 @@ public function getDefaultOperations(EntityInterface $entity) {
     return $operations;
   }
 
+  /**
+   * {@inheritdoc}
+   */
+  protected function ensureDestination(Url $url) {
+    // We don't want to add the destination URL here, as it means we get
+    // redirected back to the list-builder after adding/deleting menu links from
+    // a menu.
+    return $url;
+  }
+
   /**
    * {@inheritdoc}
    */
diff --git a/web/core/modules/menu_ui/tests/src/Functional/MenuUiTest.php b/web/core/modules/menu_ui/tests/src/Functional/MenuUiTest.php
index 2ded666bff89cd3ab9e338c53e098107f2c2ce64..d565e532036783f43e131f4e799a6cf675f6e8d6 100644
--- a/web/core/modules/menu_ui/tests/src/Functional/MenuUiTest.php
+++ b/web/core/modules/menu_ui/tests/src/Functional/MenuUiTest.php
@@ -85,6 +85,10 @@ protected function setUp(): void {
 
     $this->drupalPlaceBlock('page_title_block');
     $this->drupalPlaceBlock('system_menu_block:main');
+    $this->drupalPlaceBlock('local_actions_block', [
+      'region' => 'content',
+      'weight' => -100,
+    ]);
 
     $this->drupalCreateContentType(['type' => 'article', 'name' => 'Article']);
 
@@ -101,7 +105,7 @@ protected function setUp(): void {
   /**
    * Tests menu functionality using the admin and user interfaces.
    */
-  public function testMenu() {
+  public function testMenuAdministration() {
     // Log in the user.
     $this->drupalLogin($this->adminUser);
     $this->items = [];
@@ -195,6 +199,16 @@ public function addCustomMenuCRUD() {
     $menu->save();
     $this->drupalGet('admin/structure/menu/manage/' . $menu_name);
     $this->assertSession()->pageTextContains($new_label);
+
+    // Delete the custom menu via the UI to testing destination handling.
+    $this->drupalGet('admin/structure/menu');
+    $this->assertSession()->pageTextContains($new_label);
+    // Click the "Delete menu" operation in the Tools row.
+    $links = $this->xpath('//*/td[contains(text(),:menu_label)]/following::a[normalize-space()=:link_label]', [':menu_label' => $new_label, ':link_label' => 'Delete menu']);
+    $links[0]->click();
+    $this->submitForm([], 'Delete');
+    $this->assertSession()->addressEquals('admin/structure/menu');
+    $this->assertSession()->responseContains("The menu <em class=\"placeholder\">$new_label</em> has been deleted.");
   }
 
   /**
@@ -259,6 +273,7 @@ public function deleteCustomMenu() {
     $this->drupalGet("admin/structure/menu/manage/{$menu_name}/delete");
     $this->submitForm([], 'Delete');
     $this->assertSession()->statusCodeEquals(200);
+    $this->assertSession()->addressEquals("admin/structure/menu");
     $this->assertSession()->pageTextContains("The menu $label has been deleted.");
     $this->assertNull(Menu::load($menu_name), 'Custom menu was deleted');
     // Test if all menu links associated with the menu were removed from
@@ -279,11 +294,37 @@ public function deleteCustomMenu() {
    * Tests menu functionality.
    */
   public function doMenuTests() {
+    // Add a link to the tools menu first, to test cacheability metadata of the
+    // destination query string.
+    $this->drupalGet('admin/structure/menu/manage/tools');
+    $this->clickLink('Add link');
+    $link_title = $this->randomString();
+    $this->submitForm(['link[0][uri]' => '/', 'title[0][value]' => $link_title], 'Save');
+    $this->assertSession()->linkExists($link_title);
+    $this->assertSession()->addressEquals('admin/structure/menu/manage/tools');
+
+    // Test adding a menu link direct from the menus listing page.
+    $this->drupalGet('admin/structure/menu');
+    // Click the "Add link" operation in the Tools row.
+    $links = $this->xpath('//*/td[contains(text(),:menu_label)]/following::a[normalize-space()=:link_label]', [':menu_label' => 'Tools', ':link_label' => 'Add link']);
+    $links[0]->click();
+    $this->assertMatchesRegularExpression('#admin/structure/menu/manage/tools/add\?destination=(/[^/]*)*/admin/structure/menu/manage/tools$#', $this->getSession()->getCurrentUrl());
+    $link_title = $this->randomString();
+    $this->submitForm(['link[0][uri]' => '/', 'title[0][value]' => $link_title], 'Save');
+    $this->assertSession()->linkExists($link_title);
+    $this->assertSession()->addressEquals('admin/structure/menu/manage/tools');
+
     $menu_name = $this->menu->id();
 
-    // Test the 'Add link' local action.
-    $this->drupalGet(Url::fromRoute('entity.menu.edit_form', ['menu' => $menu_name]));
+    // Access the menu via the overview form to ensure it does not add a
+    // destination that breaks the user interface.
+    $this->drupalGet('admin/structure/menu');
 
+    // Select the edit menu link for our menu.
+    $links = $this->xpath('//*/td[contains(text(),:menu_label)]/following::a[normalize-space()=:link_label]', [':menu_label' => (string) $this->menu->label(), ':link_label' => 'Edit menu']);
+    $links[0]->click();
+
+    // Test the 'Add link' local action.
     $this->clickLink('Add link');
     $link_title = $this->randomString();
     $this->submitForm(['link[0][uri]' => '/', 'title[0][value]' => $link_title], 'Save');
@@ -300,6 +341,22 @@ public function doMenuTests() {
     $this->submitForm([], 'Delete');
     $this->assertSession()->addressEquals(Url::fromRoute('entity.menu.edit_form', ['menu' => $menu_name]));
 
+    // Clear the cache to ensure that recent caches aren't preventing us from
+    // seeing a broken add link.
+    $this->resetAll();
+    $this->drupalGet('admin/structure/menu');
+
+    // Select the edit menu link for our menu.
+    $links = $this->xpath('//*/td[contains(text(),:menu_label)]/following::a[normalize-space()=:link_label]', [':menu_label' => (string) $this->menu->label(), ':link_label' => 'Edit menu']);
+    $links[0]->click();
+
+    // Test the 'Add link' local action.
+    $this->clickLink('Add link');
+    $link_title = $this->randomString();
+    $this->submitForm(['link[0][uri]' => '/', 'title[0][value]' => $link_title], 'Save');
+    $this->assertSession()->linkExists($link_title);
+    $this->assertSession()->addressEquals(Url::fromRoute('entity.menu.edit_form', ['menu' => $menu_name]));
+
     // Add nodes to use as links for menu links.
     $node1 = $this->drupalCreateNode(['type' => 'article']);
     $node2 = $this->drupalCreateNode(['type' => 'article']);
diff --git a/web/core/modules/migrate_drupal/src/Plugin/migrate/EntityReferenceTranslationDeriver.php b/web/core/modules/migrate_drupal/src/Plugin/migrate/EntityReferenceTranslationDeriver.php
index 603cbbe70a5611dacdb2c5f3f60b899adbd99529..e66c8a170b0f896ed6904f088a86008ca073c584 100644
--- a/web/core/modules/migrate_drupal/src/Plugin/migrate/EntityReferenceTranslationDeriver.php
+++ b/web/core/modules/migrate_drupal/src/Plugin/migrate/EntityReferenceTranslationDeriver.php
@@ -153,7 +153,7 @@ public function getDerivativeDefinitions($base_plugin_definition) {
             'plugin' => 'sub_process',
             'source' => $field_name,
             'process' => [
-              'target_id' => [
+              'translation_target_id' => [
                 [
                   'plugin' => 'migration_lookup',
                   'source' => 'target_id',
@@ -162,13 +162,22 @@ public function getDerivativeDefinitions($base_plugin_definition) {
                 ],
                 [
                   'plugin' => 'skip_on_empty',
-                  'method' => 'row',
+                  'method' => 'process',
                 ],
                 [
                   'plugin' => 'extract',
                   'index' => [0],
                 ],
               ],
+              'target_id' => [
+                [
+                  'plugin' => 'null_coalesce',
+                  'source' => [
+                    '@translation_target_id',
+                    'target_id',
+                  ],
+                ],
+              ],
             ],
           ];
 
diff --git a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/DrupalSqlBase.php b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/DrupalSqlBase.php
index 54cce94105510b77a65e34627af9892a5cafadf6..428bfd2b42cdbb37ff76b0cf6e99df7dc0617ed5 100644
--- a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/DrupalSqlBase.php
+++ b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/DrupalSqlBase.php
@@ -22,7 +22,8 @@
  *
  * For a full list, refer to the methods of this class.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  */
diff --git a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/Variable.php b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/Variable.php
index 1b7679ea0dbc00f919799f0fd791137a269c4133..aed5245d0a4a08dda5fc18592681d7f2c0a4788e 100644
--- a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/Variable.php
+++ b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/Variable.php
@@ -7,7 +7,7 @@
 use Drupal\migrate\Plugin\MigrationInterface;
 
 /**
- * Drupal variable source from database.
+ * Drupal 6/7 variable source from database.
  *
  * This source class fetches variables from the source Drupal database.
  * Depending on the configuration, this returns zero or a single row and as such
@@ -70,7 +70,8 @@
  *     - book_allowed_types
  * @endcode
  *
- * For additional configuration keys, refer to the parent classes:
+ * For additional configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/VariableMultiRow.php b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/VariableMultiRow.php
index 27d820fb9eb2482162ab1e2b8cee4544bd63b744..98f22edbda6787ac7a1f0f2d7bcb76950c3f75b0 100644
--- a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/VariableMultiRow.php
+++ b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/VariableMultiRow.php
@@ -5,7 +5,7 @@
 use Drupal\migrate\Row;
 
 /**
- * Multiple variables source from database.
+ * Drupal 6/7 multiple variables source from database.
  *
  * Unlike the variable source plugin, this one returns one row per
  * variable.
@@ -27,7 +27,8 @@
  * In this example the specified variables are retrieved from the source
  * database one row per variable.
  *
- * For additional configuration keys, refer to the parent classes:
+ * For additional configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d6/VariableTranslation.php b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d6/VariableTranslation.php
index eb897ee385802a0d111193ab6546cd08847c6bb3..c41bc5d2e225720656945c4a0cfee462a8c8072e 100644
--- a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d6/VariableTranslation.php
+++ b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d6/VariableTranslation.php
@@ -9,7 +9,7 @@
 use Drupal\migrate_drupal\Plugin\migrate\source\DrupalSqlBase;
 
 /**
- * Drupal i18n_variable source from database.
+ * Drupal 6 i18n_variable source from database.
  *
  * Available configuration keys:
  * - variables: (required) The list of variable translations to retrieve from
@@ -25,7 +25,8 @@
  * In this example the translations for site_offline_message variable are
  * retrieved from the source database.
  *
- * For additional configuration keys, refer to the parent classes:
+ * For additional configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d7/FieldableEntity.php b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d7/FieldableEntity.php
index dbb5ed67d6c38418c3bbaf6c1aac5541dc45a079..176749480b5478fe9ab3f35947af739c1d0eedd2 100644
--- a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d7/FieldableEntity.php
+++ b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d7/FieldableEntity.php
@@ -12,7 +12,8 @@
  * @see \Drupal\node\Plugin\migrate\source\d7\Node
  * @see \Drupal\user\Plugin\migrate\source\d7\User
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  */
@@ -21,6 +22,12 @@ abstract class FieldableEntity extends DrupalSqlBase {
   /**
    * Returns all non-deleted field instances attached to a specific entity type.
    *
+   * Typically, getFields() is used in the prepareRow method of a source plugin
+   * to get a list of all the field instances of the entity. A source plugin can
+   * then loop through the list of fields to do any other preparation before
+   * processing the row. Typically, a source plugin will use getFieldValues()
+   * to get the values of each field.
+   *
    * @param string $entity_type
    *   The entity type ID.
    * @param string|null $bundle
@@ -47,6 +54,9 @@ protected function getFields($entity_type, $bundle = NULL) {
   /**
    * Retrieves field values for a single field of a single entity.
    *
+   * Typically, getFieldValues() is used in the prepareRow method of a source
+   * plugin where the return values are placed on the row source.
+   *
    * @param string $entity_type
    *   The entity type.
    * @param string $field
diff --git a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d7/VariableTranslation.php b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d7/VariableTranslation.php
index eb71189a442a2f3c616981ce28b7cee9aefe29df..32bdfb2eb9b92f3ad1c86382cffd1387ea359c1f 100644
--- a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d7/VariableTranslation.php
+++ b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d7/VariableTranslation.php
@@ -8,7 +8,7 @@
 use Drupal\migrate_drupal\Plugin\migrate\source\DrupalSqlBase;
 
 /**
- * Gets Drupal variable_store source from database.
+ * Drupal 7 variable_store source from database.
  *
  * Available configuration keys:
  * - variables: (required) The list of variable translations to retrieve from
@@ -25,7 +25,8 @@
  * In this example the translations for site_name and site_slogan variables are
  * retrieved from the source database.
  *
- * For additional configuration keys, refer to the parent classes:
+ * For additional configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d8/Config.php b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d8/Config.php
index 5aba90f2acdf14f996f06d035f036b9e2f9a0a0b..16b545c0fc8bcb099c9257a9ec1e28f5341b3377 100644
--- a/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d8/Config.php
+++ b/web/core/modules/migrate_drupal/src/Plugin/migrate/source/d8/Config.php
@@ -42,7 +42,8 @@
  * As a result, French versions of specified configuration objects are retrieved
  * from the source database.
  *
- * For additional configuration keys, refer to the parent classes:
+ * For additional configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/migrate_drupal/tests/fixtures/drupal7.php b/web/core/modules/migrate_drupal/tests/fixtures/drupal7.php
index 2412555aaf42fcc9a2014afa0390d763ce78c139..0662f876dc5be277476145638bf45eef6d62e204 100644
--- a/web/core/modules/migrate_drupal/tests/fixtures/drupal7.php
+++ b/web/core/modules/migrate_drupal/tests/fixtures/drupal7.php
@@ -4,7 +4,7 @@
  * @file
  * A database agnostic dump for testing purposes.
  *
- * This file was generated by the Drupal 8.0 db-tools.php script.
+ * This file was generated by the Drupal 9.3.0-dev db-tools.php script.
  */
 
 use Drupal\Core\Database\Database;
@@ -4312,7 +4312,7 @@
   'storage_module' => 'field_sql_storage',
   'storage_active' => '1',
   'locked' => '0',
-  'data' => 'a:7:{s:12:"translatable";s:1:"0";s:12:"entity_types";a:0:{}s:8:"settings";a:3:{s:11:"target_type";s:4:"node";s:7:"handler";s:4:"base";s:16:"handler_settings";a:2:{s:14:"target_bundles";a:1:{s:7:"article";s:7:"article";}s:4:"sort";a:1:{s:4:"type";s:4:"none";}}}s:7:"storage";a:5:{s:4:"type";s:17:"field_sql_storage";s:8:"settings";a:0:{}s:6:"module";s:17:"field_sql_storage";s:6:"active";s:1:"1";s:7:"details";a:1:{s:3:"sql";a:2:{s:18:"FIELD_LOAD_CURRENT";a:1:{s:28:"field_data_field_reference_2";a:1:{s:9:"target_id";s:27:"field_reference_2_target_id";}}s:19:"FIELD_LOAD_REVISION";a:1:{s:32:"field_revision_field_reference_2";a:1:{s:9:"target_id";s:27:"field_reference_2_target_id";}}}}}s:12:"foreign keys";a:1:{s:4:"node";a:2:{s:5:"table";s:4:"node";s:7:"columns";a:1:{s:9:"target_id";s:3:"nid";}}}s:7:"indexes";a:1:{s:9:"target_id";a:1:{i:0;s:9:"target_id";}}s:2:"id";s:2:"39";}',
+  'data' => 'a:7:{s:12:"translatable";s:1:"0";s:12:"entity_types";a:0:{}s:8:"settings";a:3:{s:11:"target_type";s:4:"node";s:7:"handler";s:4:"base";s:16:"handler_settings";a:2:{s:14:"target_bundles";a:2:{s:7:"article";s:7:"article";s:5:"forum";s:5:"forum";}s:4:"sort";a:1:{s:4:"type";s:4:"none";}}}s:7:"storage";a:5:{s:4:"type";s:17:"field_sql_storage";s:8:"settings";a:0:{}s:6:"module";s:17:"field_sql_storage";s:6:"active";s:1:"1";s:7:"details";a:1:{s:3:"sql";a:2:{s:18:"FIELD_LOAD_CURRENT";a:1:{s:28:"field_data_field_reference_2";a:1:{s:9:"target_id";s:27:"field_reference_2_target_id";}}s:19:"FIELD_LOAD_REVISION";a:1:{s:32:"field_revision_field_reference_2";a:1:{s:9:"target_id";s:27:"field_reference_2_target_id";}}}}}s:12:"foreign keys";a:1:{s:4:"node";a:2:{s:5:"table";s:4:"node";s:7:"columns";a:1:{s:9:"target_id";s:3:"nid";}}}s:7:"indexes";a:1:{s:9:"target_id";a:1:{i:0;s:9:"target_id";}}s:2:"id";s:2:"39";}',
   'cardinality' => '1',
   'translatable' => '0',
   'deleted' => '0',
@@ -8181,7 +8181,7 @@
   'bundle' => 'article',
   'deleted' => '0',
   'entity_id' => '2',
-  'revision_id' => '2',
+  'revision_id' => '11',
   'language' => 'und',
   'delta' => '0',
   'field_node_reference_nid' => '5',
@@ -8768,7 +8768,7 @@
   'revision_id' => '11',
   'language' => 'und',
   'delta' => '0',
-  'field_reference_2_target_id' => '5',
+  'field_reference_2_target_id' => '6',
 ))
 ->values(array(
   'entity_type' => 'node',
@@ -14731,6 +14731,16 @@
   'delta' => '0',
   'field_node_reference_nid' => '5',
 ))
+->values(array(
+  'entity_type' => 'node',
+  'bundle' => 'article',
+  'deleted' => '0',
+  'entity_id' => '2',
+  'revision_id' => '11',
+  'language' => 'und',
+  'delta' => '0',
+  'field_node_reference_nid' => '5',
+))
 ->execute();
 $connection->schema()->createTable('field_revision_field_phone', array(
   'fields' => array(
@@ -15378,7 +15388,7 @@
   'revision_id' => '11',
   'language' => 'und',
   'delta' => '0',
-  'field_reference_2_target_id' => '5',
+  'field_reference_2_target_id' => '6',
 ))
 ->values(array(
   'entity_type' => 'node',
@@ -59872,6 +59882,10 @@
   'name' => 'language_content_type_article',
   'value' => 's:1:"2";',
 ))
+->values(array(
+  'name' => 'language_content_type_a_thirty_two_character_type_name',
+  'value' => 's:1:"1";',
+))
 ->values(array(
   'name' => 'language_content_type_blog',
   'value' => 's:1:"2";',
diff --git a/web/core/modules/migrate_drupal/tests/src/Kernel/d7/FollowUpMigrationsTest.php b/web/core/modules/migrate_drupal/tests/src/Kernel/d7/FollowUpMigrationsTest.php
index 94ef1d68f715b2252929090d62f1da4bb7a92a75..b67b3f335819ee5a37026167bcfab9a519364798 100644
--- a/web/core/modules/migrate_drupal/tests/src/Kernel/d7/FollowUpMigrationsTest.php
+++ b/web/core/modules/migrate_drupal/tests/src/Kernel/d7/FollowUpMigrationsTest.php
@@ -51,8 +51,6 @@ protected function setUp(): void {
       'language',
       'd7_language_content_settings',
       'd7_taxonomy_vocabulary',
-      'd7_node',
-      'd7_node_translation',
     ]);
   }
 
@@ -70,12 +68,16 @@ protected function getFileMigrationInfo() {
 
   /**
    * Tests entity reference translations.
+   *
+   * @dataProvider providerTestEntityReferenceTranslations
    */
-  public function testEntityReferenceTranslations() {
+  public function testEntityReferenceTranslations($node_migrations) {
+    $this->executeMigrations($node_migrations);
+
     // Test the entity reference field before the follow-up migrations.
     $node = Node::load(2);
     $this->assertSame('5', $node->get('field_reference')->target_id);
-    $this->assertSame('5', $node->get('field_reference_2')->target_id);
+    $this->assertSame('6', $node->get('field_reference_2')->target_id);
     $translation = $node->getTranslation('is');
     $this->assertSame('4', $translation->get('field_reference')->target_id);
     $this->assertSame('4', $translation->get('field_reference_2')->target_id);
@@ -99,7 +101,7 @@ public function testEntityReferenceTranslations() {
     // Test the entity reference field after the follow-up migrations.
     $node = Node::load(2);
     $this->assertSame('4', $node->get('field_reference')->target_id);
-    $this->assertSame('4', $node->get('field_reference_2')->target_id);
+    $this->assertSame('6', $node->get('field_reference_2')->target_id);
     $translation = $node->getTranslation('is');
     $this->assertSame('4', $translation->get('field_reference')->target_id);
     $this->assertSame('4', $translation->get('field_reference_2')->target_id);
@@ -115,4 +117,18 @@ public function testEntityReferenceTranslations() {
     $this->assertSame('2', $user->get('field_reference')->target_id);
   }
 
+  /**
+   * Data provider for testEntityReferenceTranslations().
+   */
+  public function providerTestEntityReferenceTranslations() {
+    return [
+      [
+        ['d7_node', 'd7_node_translation'],
+      ],
+      [
+        ['d7_node_complete'],
+      ],
+    ];
+  }
+
 }
diff --git a/web/core/modules/migrate_drupal_ui/tests/src/Functional/d7/Upgrade7Test.php b/web/core/modules/migrate_drupal_ui/tests/src/Functional/d7/Upgrade7Test.php
index 2652e6db235a4761b0a29532397689427ebb2e81..b29adb07a8548382529d3ede841d4057b85bd97d 100644
--- a/web/core/modules/migrate_drupal_ui/tests/src/Functional/d7/Upgrade7Test.php
+++ b/web/core/modules/migrate_drupal_ui/tests/src/Functional/d7/Upgrade7Test.php
@@ -89,7 +89,7 @@ protected function getEntityCounts() {
       'file' => 3,
       'filter_format' => 7,
       'image_style' => 7,
-      'language_content_settings' => 22,
+      'language_content_settings' => 24,
       'node' => 7,
       'node_type' => 8,
       'rdf_mapping' => 8,
@@ -237,7 +237,7 @@ public function testUpgradeAndIncremental() {
   protected function assertFollowUpMigrationResults() {
     $node = Node::load(2);
     $this->assertSame('4', $node->get('field_reference')->target_id);
-    $this->assertSame('4', $node->get('field_reference_2')->target_id);
+    $this->assertSame('6', $node->get('field_reference_2')->target_id);
     $translation = $node->getTranslation('is');
     $this->assertSame('4', $translation->get('field_reference')->target_id);
     $this->assertSame('4', $translation->get('field_reference_2')->target_id);
diff --git a/web/core/modules/node/src/Plugin/migrate/source/d6/Node.php b/web/core/modules/node/src/Plugin/migrate/source/d6/Node.php
index 4fe426d890d7776bbc5a0fd6e05a192034b273ab..e120ef9cc671aa9e950a38624f6244dc76d2eabd 100644
--- a/web/core/modules/node/src/Plugin/migrate/source/d6/Node.php
+++ b/web/core/modules/node/src/Plugin/migrate/source/d6/Node.php
@@ -297,6 +297,10 @@ protected function getFieldData(array $field, Row $node) {
 
     if (isset($query)) {
       $columns = array_keys($field['db_columns']);
+      // If there are no columns then there are no values to return.
+      if (empty($columns)) {
+        return [];
+      }
 
       // Add every column in the field's schema.
       foreach ($columns as $column) {
diff --git a/web/core/modules/node/src/Plugin/migrate/source/d7/NodeEntityTranslation.php b/web/core/modules/node/src/Plugin/migrate/source/d7/NodeEntityTranslation.php
index ca61c6c563a05013346544e064c217629c391953..d9ea90c3739d07da0d15eb9f5e33e3031a4a6a9f 100644
--- a/web/core/modules/node/src/Plugin/migrate/source/d7/NodeEntityTranslation.php
+++ b/web/core/modules/node/src/Plugin/migrate/source/d7/NodeEntityTranslation.php
@@ -6,7 +6,7 @@
 use Drupal\migrate_drupal\Plugin\migrate\source\d7\FieldableEntity;
 
 /**
- * Provides Drupal 7 node entity translations source plugin.
+ * Drupal 7 node entity translations source from database.
  *
  * Available configuration keys:
  * - node_type: The node_types to get from the source - can be a string or
diff --git a/web/core/modules/node/tests/src/Kernel/Migrate/d7/MigrateNodeCompleteTest.php b/web/core/modules/node/tests/src/Kernel/Migrate/d7/MigrateNodeCompleteTest.php
index 1d366d34144180b08ecdcfa9f3db93c9cbf36dc0..2c3acd8cf24c32d506e3fe2f75c02fea8c847b7b 100644
--- a/web/core/modules/node/tests/src/Kernel/Migrate/d7/MigrateNodeCompleteTest.php
+++ b/web/core/modules/node/tests/src/Kernel/Migrate/d7/MigrateNodeCompleteTest.php
@@ -4,6 +4,7 @@
 
 use Drupal\migrate\MigrateExecutable;
 use Drupal\migrate_drupal\NodeMigrateType;
+use Drupal\node\Entity\Node;
 use Drupal\node\NodeInterface;
 use Drupal\Tests\file\Kernel\Migrate\d7\FileMigrationSetupTrait;
 use Drupal\Tests\migrate_drupal\Kernel\d7\MigrateDrupal7TestBase;
@@ -136,6 +137,15 @@ public function testNodeCompleteMigration() {
       $this->assertSame('Bob', $revision->field_user_reference[0]->entity->getAccountName());
     }
 
+    // Test the translated node reference in the latest revision of node 2. This
+    // references the legacy site node 4 instead of node 2. The reference is
+    // fixed by the followup migrations, 'd7_entity_reference_translation' and
+    // tested in \Drupal\Tests\migrate_drupal\Kernel\d7\FollowUpMigrationsTest.
+    $node = Node::load(2);
+    $this->assertSame('6', $node->get('field_reference_2')->target_id);
+    $translation = $node->getTranslation('is');
+    $this->assertSame('4', $translation->get('field_reference_2')->target_id);
+
     // Test the order in multi-value fields.
     $revision = $this->nodeStorage->loadRevision(1);
     $this->assertSame([
diff --git a/web/core/modules/node/tests/src/Kernel/Plugin/migrate/source/d6/NodeTest.php b/web/core/modules/node/tests/src/Kernel/Plugin/migrate/source/d6/NodeTest.php
index 9a78d61da3ff602041afaa898289197114886687..c6f72ee3eccc5eec6b7858c3e4e23cf3f676ed6f 100644
--- a/web/core/modules/node/tests/src/Kernel/Plugin/migrate/source/d6/NodeTest.php
+++ b/web/core/modules/node/tests/src/Kernel/Plugin/migrate/source/d6/NodeTest.php
@@ -38,6 +38,18 @@ public function providerSource() {
         'active' => '1',
         'locked' => '0',
       ],
+      [
+        'field_name' => 'field_test_empty_db_columns',
+        'type' => 'empty_db_columns',
+        'global_settings' => 'a:0:{}',
+        'required' => '0',
+        'multiple' => '0',
+        'db_storage' => '1',
+        'module' => 'empty_db_columns',
+        'db_columns' => 'a:0:{}',
+        'active' => '1',
+        'locked' => '0',
+      ],
     ];
     $tests[0]['source_data']['content_node_field_instance'] = [
       [
@@ -52,6 +64,18 @@ public function providerSource() {
         'widget_module' => 'number',
         'widget_active' => '1',
       ],
+      [
+        'field_name' => 'field_test_empty_db_columns',
+        'type_name' => 'story',
+        'weight' => '33',
+        'label' => 'Empty db_columns Field',
+        'widget_type' => 'empty_db_columns',
+        'widget_settings' => 'a:0:{}',
+        'display_settings' => 'a:0:{}',
+        'description' => 'An example field with empty db_columns.',
+        'widget_module' => 'empty_db_columns',
+        'widget_active' => '1',
+      ],
     ];
     $tests[0]['source_data']['content_type_story'] = [
       [
diff --git a/web/core/modules/system/tests/modules/render_attached_test/src/Controller/RenderAttachedTestController.php b/web/core/modules/system/tests/modules/render_attached_test/src/Controller/RenderAttachedTestController.php
index 1bbf8ebe65db07d8b133cdd3e0f268d702b60a86..955e3daa221de4d3d6c948c67325d115125ad094 100644
--- a/web/core/modules/system/tests/modules/render_attached_test/src/Controller/RenderAttachedTestController.php
+++ b/web/core/modules/system/tests/modules/render_attached_test/src/Controller/RenderAttachedTestController.php
@@ -80,6 +80,7 @@ public function htmlHeaderLink() {
     $render['#attached']['html_head_link'][] = [['href' => '/foo?bar=<baz>&baz=false', 'rel' => 'alternate'], TRUE];
     $render['#attached']['html_head_link'][] = [['href' => '/not-added-to-http-headers', 'rel' => 'alternate'], FALSE];
     $render['#attached']['html_head_link'][] = [['href' => '/foo/bar', 'hreflang' => 'nl', 'rel' => 'alternate'], TRUE];
+    $render['#attached']['html_head_link'][] = [['href' => '/foo/bar', 'hreflang' => 'de', 'rel' => 'alternate'], TRUE];
     return $render;
   }
 
diff --git a/web/core/modules/system/tests/src/Functional/Render/HtmlResponseAttachmentsTest.php b/web/core/modules/system/tests/src/Functional/Render/HtmlResponseAttachmentsTest.php
index f07424b455e31f0410a28c61bc95cd9452c5c9b4..9921fe04a9001f34fc563a55c56a0fad39514630 100644
--- a/web/core/modules/system/tests/src/Functional/Render/HtmlResponseAttachmentsTest.php
+++ b/web/core/modules/system/tests/src/Functional/Render/HtmlResponseAttachmentsTest.php
@@ -65,8 +65,13 @@ public function testAttachments() {
     $expected_link_headers = [
       '</foo?bar=&lt;baz&gt;&amp;baz=false>; rel="alternate"',
       '</foo/bar>; hreflang="nl"; rel="alternate"',
+      '</foo/bar>; hreflang="de"; rel="alternate"',
     ];
     $this->assertEquals($expected_link_headers, $this->getSession()->getResponseHeaders()['Link']);
+
+    // Check that duplicate alternate URLs with different hreflangs are allowed.
+    $test_link = $this->xpath('//head/link[@rel="alternate"][@href="/foo/bar"]');
+    $this->assertEquals(2, count($test_link), 'Duplicate alternate URLs are allowed.');
   }
 
   /**
diff --git a/web/core/modules/taxonomy/src/Plugin/migrate/source/d6/TermLocalizedTranslation.php b/web/core/modules/taxonomy/src/Plugin/migrate/source/d6/TermLocalizedTranslation.php
index 6fd6a1ce4e057a8dbfa4e802c7a86fee1b8caeb7..57397c62bca9ec69c85dc5cb70167acd0a9f2168 100644
--- a/web/core/modules/taxonomy/src/Plugin/migrate/source/d6/TermLocalizedTranslation.php
+++ b/web/core/modules/taxonomy/src/Plugin/migrate/source/d6/TermLocalizedTranslation.php
@@ -5,7 +5,7 @@
 use Drupal\migrate\Row;
 
 /**
- * Drupal 6 i18n taxonomy terms from source database.
+ * Drupal 6 i18n taxonomy terms source from database.
  *
  * For available configuration keys, refer to the parent classes.
  *
diff --git a/web/core/modules/taxonomy/src/Plugin/migrate/source/d6/VocabularyTranslation.php b/web/core/modules/taxonomy/src/Plugin/migrate/source/d6/VocabularyTranslation.php
index 294da00e31874c1e4a27aae686b89109d26bdfcd..617daf7e4730dbf363e707a4bf76778a1ebda510 100644
--- a/web/core/modules/taxonomy/src/Plugin/migrate/source/d6/VocabularyTranslation.php
+++ b/web/core/modules/taxonomy/src/Plugin/migrate/source/d6/VocabularyTranslation.php
@@ -6,7 +6,7 @@
 use Drupal\migrate_drupal\Plugin\migrate\source\DrupalSqlBase;
 
 /**
- * Drupal 6 i18n vocabulary translations from source database.
+ * Drupal 6 i18n vocabulary translations source from database.
  *
  * For available configuration keys, refer to the parent classes.
  *
diff --git a/web/core/modules/taxonomy/src/Plugin/migrate/source/d7/TermLocalizedTranslation.php b/web/core/modules/taxonomy/src/Plugin/migrate/source/d7/TermLocalizedTranslation.php
index a06ec0d138c60446e4f25f1476121c33e86aa801..43d5cfaaf4a22a5d4e930bad9d548f1ccc1e62ec 100644
--- a/web/core/modules/taxonomy/src/Plugin/migrate/source/d7/TermLocalizedTranslation.php
+++ b/web/core/modules/taxonomy/src/Plugin/migrate/source/d7/TermLocalizedTranslation.php
@@ -6,7 +6,7 @@
 use Drupal\migrate\Row;
 
 /**
- * Drupal 7 i18n taxonomy terms from source database.
+ * Drupal 7 i18n taxonomy terms source from database.
  *
  * For available configuration keys, refer to the parent classes.
  *
diff --git a/web/core/modules/taxonomy/src/Plugin/migrate/source/d7/TermTranslation.php b/web/core/modules/taxonomy/src/Plugin/migrate/source/d7/TermTranslation.php
index 50fb58161ab1b2f198881dbfe38e7e32105433ac..c58dba0731ee6ee51cd5d5e27e96de8927596911 100644
--- a/web/core/modules/taxonomy/src/Plugin/migrate/source/d7/TermTranslation.php
+++ b/web/core/modules/taxonomy/src/Plugin/migrate/source/d7/TermTranslation.php
@@ -5,7 +5,7 @@
 use Drupal\migrate\Row;
 
 /**
- * Drupal 7 i18n taxonomy terms from source database.
+ * Drupal 7 i18n taxonomy terms source from database.
  *
  * For available configuration keys, refer to the parent classes.
  *
@@ -28,9 +28,17 @@ public function query() {
     if ($this->database->schema()->fieldExists('taxonomy_term_data', 'language')) {
       $query->addField('td', 'language', 'td_language');
     }
+    // Get data when the i18n_mode column exists and it is not the Drupal 7
+    // value I18N_MODE_NONE or I18N_MODE_LOCALIZE. Otherwise, return no data.
+    // @see https://git.drupalcode.org/project/i18n/-/blob/7.x-1.x/i18n.module#L26
     if ($this->database->schema()->fieldExists('taxonomy_vocabulary', 'i18n_mode')) {
       $query->addField('tv', 'i18n_mode');
+      $query->condition('tv.i18n_mode', ['0', '1'], 'NOT IN');
     }
+    else {
+      $query->alwaysFalse();
+    }
+
     return $query;
   }
 
diff --git a/web/core/modules/taxonomy/tests/src/Kernel/Plugin/migrate/source/d7/TermTranslationTest.php b/web/core/modules/taxonomy/tests/src/Kernel/Plugin/migrate/source/d7/TermTranslationTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..a218b3ec3ee783ea418ba742a91a846672e73b93
--- /dev/null
+++ b/web/core/modules/taxonomy/tests/src/Kernel/Plugin/migrate/source/d7/TermTranslationTest.php
@@ -0,0 +1,361 @@
+<?php
+
+namespace Drupal\Tests\taxonomy\Kernel\Plugin\migrate\source\d7;
+
+/**
+ * Tests D7 i18n term localized source plugin.
+ *
+ * @covers \Drupal\taxonomy\Plugin\migrate\source\d7\TermTranslation
+ * @group taxonomy
+ */
+class TermTranslationTest extends TermTest {
+
+  /**
+   * {@inheritdoc}
+   */
+  protected static $modules = ['taxonomy', 'migrate_drupal'];
+
+  /**
+   * {@inheritdoc}
+   */
+  public function providerSource() {
+    $tests = [];
+
+    // Ignore i18_modes 0 and 1, get i18n_mode 4.
+    $tests[0]['source_data']['taxonomy_term_data'] = [
+      [
+        'tid' => 1,
+        'vid' => 5,
+        'name' => 'fr - name 1',
+        'description' => 'desc 1',
+        'weight' => 0,
+        'is_container' => FALSE,
+        'language' => 'fr',
+        'i18n_tsid' => '1',
+      ],
+      [
+        'tid' => 2,
+        'vid' => 5,
+        'name' => 'name 2',
+        'description' => 'desc 2',
+        'weight' => 0,
+        'is_container' => TRUE,
+        'language' => 'en',
+        'i18n_tsid' => '1',
+      ],
+      [
+        'tid' => 3,
+        'vid' => 6,
+        'name' => 'name 3',
+        'description' => 'desc 3',
+        'weight' => 0,
+        'is_container' => FALSE,
+        'language' => '',
+        'i18n_tsid' => '',
+      ],
+      [
+        'tid' => 4,
+        'vid' => 5,
+        'name' => 'is - name 4',
+        'description' => 'desc 4',
+        'weight' => 1,
+        'is_container' => FALSE,
+        'language' => 'is',
+        'i18n_tsid' => '1',
+      ],
+      [
+        'tid' => 5,
+        'vid' => 6,
+        'name' => 'name 5',
+        'description' => 'desc 5',
+        'weight' => 1,
+        'is_container' => FALSE,
+        'language' => '',
+        'i18n_tsid' => '',
+      ],
+      [
+        'tid' => 6,
+        'vid' => 6,
+        'name' => 'name 6',
+        'description' => 'desc 6',
+        'weight' => 0,
+        'is_container' => TRUE,
+        'language' => '',
+        'i18n_tsid' => '',
+      ],
+      [
+        'tid' => 7,
+        'vid' => 7,
+        'name' => 'is - captains',
+        'description' => 'desc 7',
+        'weight' => 0,
+        'is_container' => TRUE,
+        'language' => 'is',
+        'i18n_tsid' => '',
+      ],
+    ];
+    $tests[0]['source_data']['taxonomy_term_hierarchy'] = [
+      [
+        'tid' => 1,
+        'parent' => 0,
+      ],
+      [
+        'tid' => 2,
+        'parent' => 0,
+      ],
+      [
+        'tid' => 3,
+        'parent' => 0,
+      ],
+      [
+        'tid' => 4,
+        'parent' => 1,
+      ],
+      [
+        'tid' => 5,
+        'parent' => 2,
+      ],
+      [
+        'tid' => 6,
+        'parent' => 3,
+      ],
+      [
+        'tid' => 6,
+        'parent' => 2,
+      ],
+      [
+        'tid' => 7,
+        'parent' => 0,
+      ],
+    ];
+
+    $tests[0]['source_data']['taxonomy_vocabulary'] = [
+      [
+        'vid' => 3,
+        'machine_name' => 'foo',
+        'language' => 'und',
+        'i18n_mode' => '0',
+      ],
+      [
+        'vid' => 5,
+        'machine_name' => 'tags',
+        'language' => 'und',
+        'i18n_mode' => '4',
+      ],
+      [
+        'vid' => 6,
+        'machine_name' => 'categories',
+        'language' => 'is',
+        'i18n_mode' => '1',
+      ],
+    ];
+
+    $tests[0]['source_data']['field_config'] = [
+      [
+        'id' => '3',
+        'translatable' => '0',
+      ],
+      [
+        'id' => '4',
+        'translatable' => '1',
+      ],
+      [
+        'id' => '5',
+        'translatable' => '1',
+      ],
+    ];
+    $tests[0]['source_data']['field_config_instance'] = [
+      [
+        'id' => '2',
+        'field_id' => 3,
+        'field_name' => 'field_term_field',
+        'entity_type' => 'taxonomy_term',
+        'bundle' => 'tags',
+        'data' => 'a:0:{}',
+        'deleted' => 0,
+      ],
+      [
+        'id' => '3',
+        'field_id' => 3,
+        'field_name' => 'field_term_field',
+        'entity_type' => 'taxonomy_term',
+        'bundle' => 'categories',
+        'data' => 'a:0:{}',
+        'deleted' => 0,
+      ],
+      [
+        'id' => '4',
+        'field_id' => '4',
+        'field_name' => 'name_field',
+        'entity_type' => 'taxonomy_term',
+        'bundle' => 'tags',
+        'data' => 'a:0:{}',
+        'deleted' => '0',
+      ],
+      [
+        'id' => '5',
+        'field_id' => '5',
+        'field_name' => 'description_field',
+        'entity_type' => 'taxonomy_term',
+        'bundle' => 'tags',
+        'data' => 'a:0:{}',
+        'deleted' => '0',
+      ],
+    ];
+    $tests[0]['source_data']['field_data_field_term_field'] = [
+      [
+        'entity_type' => 'taxonomy_term',
+        'bundle' => 'tags',
+        'deleted' => 0,
+        'entity_id' => 1,
+        'delta' => 0,
+      ],
+      [
+        'entity_type' => 'taxonomy_term',
+        'bundle' => 'categories',
+        'deleted' => 0,
+        'entity_id' => 1,
+        'delta' => 0,
+      ],
+    ];
+    $tests[0]['source_data']['field_data_name_field'] = [
+      [
+        'entity_type' => 'taxonomy_term',
+        'bundle' => 'tags',
+        'deleted' => '0',
+        'entity_id' => '1',
+        'revision_id' => '1',
+        'language' => 'und',
+        'delta' => '0',
+        'name_field_value' => 'fr - name 1',
+        'name_field_format' => NULL,
+      ],
+      [
+        'entity_type' => 'taxonomy_term',
+        'bundle' => 'tags',
+        'deleted' => '0',
+        'entity_id' => '4',
+        'revision_id' => '4',
+        'language' => 'und',
+        'delta' => '0',
+        'name_field_value' => 'is - name 4',
+        'name_field_format' => NULL,
+      ],
+    ];
+    $tests[0]['source_data']['field_data_description_field'] = [
+      [
+        'entity_type' => 'taxonomy_term',
+        'bundle' => 'tags',
+        'deleted' => '0',
+        'entity_id' => '1',
+        'revision_id' => '1',
+        'language' => 'und',
+        'delta' => '0',
+        'description_field_value' => 'desc 1',
+        'description_field_format' => NULL,
+      ],
+      [
+        'entity_type' => 'taxonomy_term',
+        'bundle' => 'tags',
+        'deleted' => '0',
+        'entity_id' => '4',
+        'revision_id' => '4',
+        'language' => 'und',
+        'delta' => '0',
+        'description_field_value' => 'desc 4',
+        'description_field_format' => NULL,
+      ],
+    ];
+
+    // The expected results.
+    $tests[0]['expected_data'] = [
+      [
+        'tid' => 1,
+        'vid' => 5,
+        'name' => 'fr - name 1',
+        'description' => 'desc 1',
+        'weight' => 0,
+        'is_container' => '',
+        'language' => 'fr',
+        'i18n_tsid' => '1',
+        'machine_name' => 'tags',
+        'i18n_mode' => '4',
+        'td_language' => 'fr',
+        'tv_i18n_mode' => '4',
+      ],
+      [
+        'tid' => 2,
+        'vid' => 5,
+        'name' => 'name 2',
+        'description' => 'desc 2',
+        'weight' => 0,
+        'is_container' => '',
+        'language' => 'en',
+        'i18n_tsid' => '1',
+        'machine_name' => 'tags',
+        'i18n_mode' => '4',
+        'td_language' => 'en',
+        'tv_i18n_mode' => '4',
+      ],
+      [
+        'tid' => 4,
+        'vid' => 5,
+        'name' => 'is - name 4',
+        'description' => 'desc 4',
+        'weight' => 1,
+        'is_container' => '',
+        'language' => 'is',
+        'i18n_tsid' => '1',
+        'machine_name' => 'tags',
+        'i18n_mode' => '4',
+        'td_language' => 'is',
+        'tv_i18n_mode' => '4',
+      ],
+    ];
+
+    $tests[0]['expected_count'] = NULL;
+    // Get translations for the tags bundle.
+    $tests[0]['configuration']['bundle'] = ['tags'];
+
+    // Ignore i18_modes 0. get i18n_mode 2 and 4.
+    $tests[1] = $tests[0];
+    // Change a vocabulary to using fixed translation.
+    $tests[1]['source_data']['taxonomy_vocabulary'][2] = [
+      'vid' => 7,
+      'machine_name' => 'categories',
+      'language' => 'is',
+      'i18n_mode' => '2',
+    ];
+
+    // Add the term with fixed translation.
+    $tests[1]['expected_data'][] = [
+      'tid' => 7,
+      'vid' => 7,
+      'name' => 'is - captains',
+      'description' => 'desc 7',
+      'weight' => 0,
+      'is_container' => '',
+      'language' => 'is',
+      'i18n_tsid' => '',
+      'machine_name' => 'categories',
+      'i18n_mode' => '2',
+      'td_language' => 'is',
+      'tv_i18n_mode' => '2',
+    ];
+
+    $tests[1]['expected_count'] = NULL;
+    $tests[1]['configuration']['bundle'] = NULL;
+
+    // No data returned when there is no i18n_mode column.
+    $tests[2] = [];
+    $tests[2]['source_data'] = $tests[0]['source_data'];
+    foreach ($tests[2]['source_data']['taxonomy_vocabulary'] as &$table) {
+      unset($table['i18n_mode']);
+    }
+    $tests[2]['expected_data'] = [0];
+    $tests[2]['expected_count'] = 0;
+
+    return $tests;
+  }
+
+}
diff --git a/web/core/modules/text/src/Plugin/migrate/field/d7/TextField.php b/web/core/modules/text/src/Plugin/migrate/field/d7/TextField.php
index 2d2396ed87070ff921d494ff089a8e63d7cc0b54..21bcd74d8710eb7af5dcd8650bd39f753252a182 100644
--- a/web/core/modules/text/src/Plugin/migrate/field/d7/TextField.php
+++ b/web/core/modules/text/src/Plugin/migrate/field/d7/TextField.php
@@ -34,7 +34,7 @@ public function getFieldFormatterType(Row $row) {
         break;
 
       case 'string_long':
-        $formatter_type = str_replace('text_default', 'basic_string', $formatter_type);
+        $formatter_type = str_replace(['text_default', 'text_plain'], 'basic_string', $formatter_type);
         break;
     }
 
diff --git a/web/core/modules/text/tests/src/Unit/Plugin/migrate/field/d7/TextFieldTest.php b/web/core/modules/text/tests/src/Unit/Plugin/migrate/field/d7/TextFieldTest.php
index 6923a2ec5e960f9d1e51ff6620139da42c6db319..71adb4e59d12b39a842b7a595445e3f54c1462b6 100644
--- a/web/core/modules/text/tests/src/Unit/Plugin/migrate/field/d7/TextFieldTest.php
+++ b/web/core/modules/text/tests/src/Unit/Plugin/migrate/field/d7/TextFieldTest.php
@@ -25,12 +25,13 @@ protected function setUp(): void {
   }
 
   /**
-   * Data provider for getFieldFormatterType().
+   * Data provider for testGetFieldFormatterType().
    */
   public function getFieldFormatterTypeProvider() {
     return [
       ['text', 'text_plain', 'string'],
       ['text_long', 'text_default', 'basic_string'],
+      ['text_long', 'text_plain', 'basic_string'],
     ];
   }
 
@@ -39,7 +40,7 @@ public function getFieldFormatterTypeProvider() {
    * @covers ::getFieldType
    * @dataProvider getFieldFormatterTypeProvider
    */
-  public function testGetFieldType($type, $formatter_type, $expected) {
+  public function testGetFieldFormatterType($type, $formatter_type, $expected) {
     $row = new Row();
     $row->setSourceProperty('type', $type);
     $row->setSourceProperty('formatter/type', $formatter_type);
diff --git a/web/core/modules/toolbar/tests/modules/toolbar_test/toolbar_test.module b/web/core/modules/toolbar/tests/modules/toolbar_test/toolbar_test.module
index 1f919a1e0b3f5d30493378ad097b9b572138fd01..cbc014054d3c5075601ec2ad65ae63d15fda7403 100644
--- a/web/core/modules/toolbar/tests/modules/toolbar_test/toolbar_test.module
+++ b/web/core/modules/toolbar/tests/modules/toolbar_test/toolbar_test.module
@@ -48,3 +48,14 @@ function toolbar_test_toolbar() {
 
   return $items;
 }
+
+/**
+ * Implements hook_preprocess_HOOK().
+ */
+function toolbar_test_preprocess_menu(&$variables) {
+  // All the standard hook_theme variables should be populated when the
+  // Toolbar module is rendering a menu.
+  foreach (['menu_name', 'items', 'attributes'] as $variable) {
+    $variables[$variable];
+  }
+}
diff --git a/web/core/modules/toolbar/toolbar.module b/web/core/modules/toolbar/toolbar.module
index 483a30b76ae22f3a9441f40741ac550bb0091be3..5451dcb751567d27c56b1cfda9f7937ce9f9d144 100644
--- a/web/core/modules/toolbar/toolbar.module
+++ b/web/core/modules/toolbar/toolbar.module
@@ -41,7 +41,7 @@ function toolbar_theme($existing, $type, $theme, $path) {
   ];
   $items['menu__toolbar'] = [
     'base hook' => 'menu',
-    'variables' => ['items' => [], 'attributes' => []],
+    'variables' => ['menu_name' => NULL, 'items' => [], 'attributes' => []],
   ];
 
   return $items;
diff --git a/web/core/modules/user/src/Plugin/migrate/source/ProfileField.php b/web/core/modules/user/src/Plugin/migrate/source/ProfileField.php
index ce40c399afb1d80e4fc0b9d6d5c536a5f6b2a493..db87a26fc15e816ad529480b78cbff6b30bceb89 100644
--- a/web/core/modules/user/src/Plugin/migrate/source/ProfileField.php
+++ b/web/core/modules/user/src/Plugin/migrate/source/ProfileField.php
@@ -7,9 +7,10 @@
 use Drupal\migrate\Row;
 
 /**
- * Profile field source from database.
+ * Drupal 6/7 profile field source from database.
+ *
+ * For available configuration keys, refer to the parent classes.
  *
- * For available configuration keys, refer to the parent classes:
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/user/src/Plugin/migrate/source/UserPictureInstance.php b/web/core/modules/user/src/Plugin/migrate/source/UserPictureInstance.php
index 591cc441e5a30836203444a4c8cc0358b03d9331..cd4f0b00ac1c125e7c153bbb1b20a8a9424d8612 100644
--- a/web/core/modules/user/src/Plugin/migrate/source/UserPictureInstance.php
+++ b/web/core/modules/user/src/Plugin/migrate/source/UserPictureInstance.php
@@ -6,9 +6,10 @@
 use Drupal\migrate\Plugin\migrate\source\DummyQueryTrait;
 
 /**
- * User picture field instance source from database.
+ * Drupal 6/7 user picture field instance source from database.
+ *
+ * For available configuration keys, refer to the parent classes.
  *
- * For available configuration keys, refer to the parent classes:
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/user/src/Plugin/migrate/source/d6/ProfileFieldOptionTranslation.php b/web/core/modules/user/src/Plugin/migrate/source/d6/ProfileFieldOptionTranslation.php
index b74fb5e260e1fac637f703d6bb1f5b2288957cf3..41c384876521aa548f0649b43882f198d8455e52 100644
--- a/web/core/modules/user/src/Plugin/migrate/source/d6/ProfileFieldOptionTranslation.php
+++ b/web/core/modules/user/src/Plugin/migrate/source/d6/ProfileFieldOptionTranslation.php
@@ -7,7 +7,8 @@
 /**
  * Drupal 6 i18n profile field option labels source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/user/src/Plugin/migrate/source/d6/ProfileFieldValues.php b/web/core/modules/user/src/Plugin/migrate/source/d6/ProfileFieldValues.php
index f73aa059d44a0fa5e48a27a823166ba5e616ddf5..8445be5cc3f8ab898335790aefad3a857c345618 100644
--- a/web/core/modules/user/src/Plugin/migrate/source/d6/ProfileFieldValues.php
+++ b/web/core/modules/user/src/Plugin/migrate/source/d6/ProfileFieldValues.php
@@ -8,7 +8,8 @@
 /**
  * Drupal 6 profile fields values source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/user/src/Plugin/migrate/source/d6/Role.php b/web/core/modules/user/src/Plugin/migrate/source/d6/Role.php
index 6be90ea8e75205e4422cefc2e60b9bcd2e65beaa..26cdc5408215a14c11c61bcba93be2690ed6ba98 100644
--- a/web/core/modules/user/src/Plugin/migrate/source/d6/Role.php
+++ b/web/core/modules/user/src/Plugin/migrate/source/d6/Role.php
@@ -8,7 +8,8 @@
 /**
  * Drupal 6 role source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/user/src/Plugin/migrate/source/d6/User.php b/web/core/modules/user/src/Plugin/migrate/source/d6/User.php
index 99cdcdebdf425bac2b94e5c1b86bf4e446c4ba7c..6c2349f21081853332627062e4eea91dc8823a44 100644
--- a/web/core/modules/user/src/Plugin/migrate/source/d6/User.php
+++ b/web/core/modules/user/src/Plugin/migrate/source/d6/User.php
@@ -8,7 +8,8 @@
 /**
  * Drupal 6 user source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/user/src/Plugin/migrate/source/d6/UserPicture.php b/web/core/modules/user/src/Plugin/migrate/source/d6/UserPicture.php
index c32a77385da647b6148963945f644046afe55d8d..cd52c7bd20c87ce40c70ad5020f1b85203784c16 100644
--- a/web/core/modules/user/src/Plugin/migrate/source/d6/UserPicture.php
+++ b/web/core/modules/user/src/Plugin/migrate/source/d6/UserPicture.php
@@ -7,7 +7,8 @@
 /**
  * Drupal 6 user picture source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/user/src/Plugin/migrate/source/d6/UserPictureFile.php b/web/core/modules/user/src/Plugin/migrate/source/d6/UserPictureFile.php
index 60e38e8e4dde1090d9690e0adf6a873f6935403d..02ba6163f6bcdd10811e1b85e840009f1e99276d 100644
--- a/web/core/modules/user/src/Plugin/migrate/source/d6/UserPictureFile.php
+++ b/web/core/modules/user/src/Plugin/migrate/source/d6/UserPictureFile.php
@@ -12,7 +12,8 @@
  * - site_path: (optional) The path to the site directory relative to Drupal
  *   root. Defaults to 'sites/default'.
  *
- * For additional configuration keys, refer to the parent classes:
+ * For additional configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/user/src/Plugin/migrate/source/d7/Role.php b/web/core/modules/user/src/Plugin/migrate/source/d7/Role.php
index 2939cd9549861464edc17999dd1450d6d6865f7d..d07842058a60e1040735053baf876ad3318b583c 100644
--- a/web/core/modules/user/src/Plugin/migrate/source/d7/Role.php
+++ b/web/core/modules/user/src/Plugin/migrate/source/d7/Role.php
@@ -8,7 +8,8 @@
 /**
  * Drupal 7 role source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/user/src/Plugin/migrate/source/d7/User.php b/web/core/modules/user/src/Plugin/migrate/source/d7/User.php
index e92ae14e5fac73ebc5926a8090455b8313f0384b..286857a6b6b8a78c77f5ef8fa11c31ecc77488d5 100644
--- a/web/core/modules/user/src/Plugin/migrate/source/d7/User.php
+++ b/web/core/modules/user/src/Plugin/migrate/source/d7/User.php
@@ -8,7 +8,8 @@
 /**
  * Drupal 7 user source from database.
  *
- * For available configuration keys, refer to the parent classes:
+ * For available configuration keys, refer to the parent classes.
+ *
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/user/src/Plugin/migrate/source/d7/UserEntityTranslation.php b/web/core/modules/user/src/Plugin/migrate/source/d7/UserEntityTranslation.php
index 067f86b6950edc179b511e4b49fdf79fff94bd34..04c6cead822e36f6bdad3c20111b89c7b1395e81 100644
--- a/web/core/modules/user/src/Plugin/migrate/source/d7/UserEntityTranslation.php
+++ b/web/core/modules/user/src/Plugin/migrate/source/d7/UserEntityTranslation.php
@@ -6,9 +6,10 @@
 use Drupal\migrate_drupal\Plugin\migrate\source\d7\FieldableEntity;
 
 /**
- * Provides Drupal 7 user entity translations source plugin.
+ * Drupal 7 user entity translations source from database.
+ *
+ * For available configuration keys, refer to the parent classes.
  *
- * For available configuration keys, refer to the parent classes:
  * @see \Drupal\migrate\Plugin\migrate\source\SqlBase
  * @see \Drupal\migrate\Plugin\migrate\source\SourcePluginBase
  *
diff --git a/web/core/modules/views/src/Plugin/views/query/Sql.php b/web/core/modules/views/src/Plugin/views/query/Sql.php
index c6200bc16a9671cf9e0e1e1bfff6d4694c858bb1..53a9ac037c68abaa1c53098583104c13be5a2b2c 100644
--- a/web/core/modules/views/src/Plugin/views/query/Sql.php
+++ b/web/core/modules/views/src/Plugin/views/query/Sql.php
@@ -326,8 +326,14 @@ public function buildOptionsForm(&$form, FormStateInterface $form_state) {
    */
   public function submitOptionsForm(&$form, FormStateInterface $form_state) {
     $element = ['#parents' => ['query', 'options', 'query_tags']];
-    $value = explode(',', NestedArray::getValue($form_state->getValues(), $element['#parents']));
-    $value = array_filter(array_map('trim', $value));
+    $value = NestedArray::getValue($form_state->getValues(), $element['#parents']);
+    // When toggling a display to override defaults or vice-versa the submit
+    // handler gets invoked twice, and we don't want to bash the values from the
+    // original call.
+    if (is_array($value)) {
+      return;
+    }
+    $value = array_filter(array_map('trim', explode(',', $value)));
     $form_state->setValueForElement($element, $value);
   }
 
diff --git a/web/core/modules/views/tests/src/Functional/Plugin/QueryOptionsTest.php b/web/core/modules/views/tests/src/Functional/Plugin/QueryOptionsTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..cb027c06e12c5f2e5bbb1aaaabc38be33b9bdb45
--- /dev/null
+++ b/web/core/modules/views/tests/src/Functional/Plugin/QueryOptionsTest.php
@@ -0,0 +1,70 @@
+<?php
+
+namespace Drupal\Tests\views\Functional\Plugin;
+
+use Drupal\Tests\views\Functional\ViewTestBase;
+
+/**
+ * Tests setting the query options.
+ *
+ * @group views
+ */
+class QueryOptionsTest extends ViewTestBase {
+
+  /**
+   * Views used by this test.
+   *
+   * @var array
+   */
+  public static $testViews = ['test_view'];
+
+  /**
+   * Modules to enable.
+   *
+   * @var array
+   */
+  protected static $modules = ['node', 'views_ui'];
+
+  /**
+   * {@inheritdoc}
+   */
+  protected $defaultTheme = 'stark';
+
+  /**
+   * Test that query overrides are stored.
+   */
+  public function testStoreQuerySettingsOverride() {
+    // Show the default display so the override selection is shown.
+    \Drupal::configFactory()->getEditable('views.settings')->set('ui.show.default_display', TRUE)->save();
+
+    $admin_user = $this->drupalCreateUser([
+      'administer views',
+      'administer site configuration',
+    ]);
+    $this->drupalLogin($admin_user);
+
+    $edit = [];
+    $this->drupalGet('admin/structure/views/view/test_view/edit');
+    $this->submitForm($edit, 'Add Page');
+
+    $this->drupalGet('admin/structure/views/nojs/display/test_view/page_1/query');
+    $this->assertSession()->checkboxNotChecked('query[options][distinct]');
+    $edit = [
+      'override[dropdown]' => 'page_1',
+      'query[options][distinct]' => 1,
+    ];
+    $this->submitForm($edit, 'Apply');
+    $this->drupalGet('admin/structure/views/nojs/display/test_view/page_1/query');
+    $this->assertSession()->checkboxChecked('query[options][distinct]');
+    $edit = [
+      'query[options][query_comment]' => 'comment',
+      'query[options][query_tags]' => 'query_tag, another_tag',
+    ];
+    $this->submitForm($edit, 'Apply');
+    $this->drupalGet('admin/structure/views/nojs/display/test_view/page_1/query');
+    $this->assertSession()->checkboxChecked('query[options][distinct]');
+    $this->assertSession()->fieldValueEquals('query[options][query_comment]', 'comment');
+    $this->assertSession()->fieldValueEquals('query[options][query_tags]', 'query_tag, another_tag');
+  }
+
+}
diff --git a/web/core/modules/views/tests/src/Kernel/Entity/EntityViewsDataTest.php b/web/core/modules/views/tests/src/Kernel/Entity/EntityViewsDataTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..6bab4cfdbdec1756c16a56aa2bf30df6f42f098a
--- /dev/null
+++ b/web/core/modules/views/tests/src/Kernel/Entity/EntityViewsDataTest.php
@@ -0,0 +1,821 @@
+<?php
+
+namespace Drupal\Tests\views\Kernel\Entity;
+
+use Drupal\Core\Cache\CacheBackendInterface;
+use Drupal\Core\Config\Entity\ConfigEntityBase;
+use Drupal\Core\Config\Entity\ConfigEntityType;
+use Drupal\Core\Entity\ContentEntityBase;
+use Drupal\Core\Entity\ContentEntityType;
+use Drupal\Core\Entity\EntityTypeInterface;
+use Drupal\Core\Field\BaseFieldDefinition;
+use Drupal\KernelTests\KernelTestBase;
+use Drupal\views\EntityViewsData;
+
+/**
+ * Tests entity views data.
+ *
+ * @coversDefaultClass \Drupal\views\EntityViewsData
+ * @group views
+ */
+class EntityViewsDataTest extends KernelTestBase {
+
+  /**
+   * The entity type manager.
+   *
+   * @var \Drupal\Core\Entity\EntityTypeManagerInterface
+   */
+  protected $entityTypeManager;
+
+  /**
+   * The base entity type definition, which some tests modify.
+   *
+   * Uses a custom class which allows changing entity keys.
+   *
+   * @var \Drupal\Tests\views\Kernel\Entity\TestEntityType
+   */
+  protected $baseEntityType;
+
+  /**
+   * The common base fields for test entity types.
+   *
+   * @var \Drupal\Core\Field\BaseFieldDefinition[]
+   */
+  protected $commonBaseFields;
+
+  /**
+   * Modules to enable.
+   *
+   * @var array
+   */
+  protected static $modules = [
+    'user',
+    'system',
+    'field',
+    'text',
+    'filter',
+  ];
+
+  /**
+   * {@inheritdoc}
+   */
+  protected function setUp(): void {
+    parent::setUp();
+
+    $this->entityTypeManager = $this->container->get('entity_type.manager');
+
+    // A common entity type definition. Tests may change this prior to passing
+    // it to setUpEntityType().
+    $this->baseEntityType = new TestEntityType([
+      // A normal entity type would have its class picked up during discovery,
+      // but as we're mocking this without an annotation we have to specify it.
+      'class' => ViewsTestEntity::class,
+      'base_table' => 'entity_test',
+      'id' => 'entity_test',
+      'label' => 'Entity test',
+      'entity_keys' => [
+        'uuid' => 'uuid',
+        'id' => 'id',
+        'langcode' => 'langcode',
+        'bundle' => 'type',
+        'revision' => 'revision_id',
+      ],
+      'handlers' => [
+        'views_data' => EntityViewsData::class,
+      ],
+      'provider' => 'entity_test',
+      'list_cache_contexts' => ['entity_test_list_cache_context'],
+    ]);
+
+    // Base fields for the test entity types.
+    $this->commonBaseFields['name'] = BaseFieldDefinition::create('string')
+      ->setLabel(t('Name'))
+      ->setDescription(t('The name of the test entity.'))
+      ->setTranslatable(TRUE)
+      ->setSetting('max_length', 32);
+
+    $this->commonBaseFields['created'] = BaseFieldDefinition::create('created')
+      ->setLabel(t('Authored on'))
+      ->setDescription(t('Time the entity was created'))
+      ->setTranslatable(TRUE);
+
+    $this->commonBaseFields['user_id'] = BaseFieldDefinition::create('entity_reference')
+      ->setLabel(t('User ID'))
+      ->setDescription(t('The ID of the associated user.'))
+      ->setSetting('target_type', 'user')
+      ->setSetting('handler', 'default')
+      // Default EntityTest entities to have the root user as the owner, to
+      // simplify testing.
+      ->setDefaultValue([0 => ['target_id' => 1]])
+      ->setTranslatable(TRUE);
+
+    // Add a description field. This example comes from the taxonomy Term
+    // entity.
+    $this->commonBaseFields['description'] = BaseFieldDefinition::create('text_long')
+      ->setLabel('Description')
+      ->setDescription('A description of the term.')
+      ->setTranslatable(TRUE);
+
+    // Add a URL field; this example is from the Comment entity.
+    $this->commonBaseFields['homepage'] = BaseFieldDefinition::create('uri')
+      ->setLabel('Homepage')
+      ->setDescription("The comment author's home page address.")
+      ->setTranslatable(TRUE)
+      ->setSetting('max_length', 255);
+
+    // A base field with cardinality > 1
+    $this->commonBaseFields['string'] = BaseFieldDefinition::create('string')
+      ->setLabel('Strong')
+      ->setTranslatable(TRUE)
+      ->setCardinality(2);
+
+    // Set up the basic 'entity_test' entity type. This is used by several
+    // tests; others customize it and the base fields.
+    $this->setUpEntityType($this->baseEntityType, $this->commonBaseFields);
+  }
+
+  /**
+   * Mocks an entity type and its base fields.
+   *
+   * This works by:
+   * - inserting the entity type definition into the entity type manager's cache
+   * - setting the base fields on the ViewsTestEntity class as a static property
+   *   for its baseFieldsDefinitions() method to use.
+   *
+   * @param \Drupal\Core\Entity\EntityTypeInterface $definition
+   *   An entity type definition to add to the entity type manager.
+   * @param \Drupal\Core\Field\BaseFieldDefinition[] $base_fields
+   *   An array of base field definitions
+   */
+  protected function setUpEntityType(EntityTypeInterface $definition, array $base_fields = []) {
+    // Replace the cache backend in the entity type manager so it returns
+    // our test entity type in addition to the existing ones.
+    $definitions = $this->entityTypeManager->getDefinitions();
+    $definitions[$definition->id()] = $definition;
+
+    $cache_backend = $this->prophesize(CacheBackendInterface::class);
+    $cache_data = new \StdClass();
+    $cache_data->data = $definitions;
+    $cache_backend->get('entity_type')->willReturn($cache_data);
+    $this->entityTypeManager->setCacheBackend($cache_backend->reveal(), 'entity_type', ['entity_types']);
+    $this->entityTypeManager->clearCachedDefinitions();
+
+    if ($base_fields) {
+      ViewsTestEntity::setMockedBaseFieldDefinitions($definition->id(), $base_fields);
+    }
+  }
+
+  /**
+   * Tests base tables.
+   */
+  public function testBaseTables() {
+    $data = $this->entityTypeManager->getHandler('entity_test', 'views_data')->getViewsData();
+
+    $this->assertEquals('entity_test', $data['entity_test']['table']['entity type']);
+    $this->assertEquals(FALSE, $data['entity_test']['table']['entity revision']);
+    $this->assertEquals('Entity test', $data['entity_test']['table']['group']);
+    $this->assertEquals('entity_test', $data['entity_test']['table']['provider']);
+
+    $this->assertEquals('id', $data['entity_test']['table']['base']['field']);
+    $this->assertEquals(['entity_test_list_cache_context'], $data['entity_test']['table']['base']['cache_contexts']);
+    $this->assertEquals('Entity test', $data['entity_test']['table']['base']['title']);
+
+    // TODO: change these to assertArrayNotHasKey().
+    $this->assertFalse(isset($data['entity_test']['table']['defaults']));
+
+    $this->assertFalse(isset($data['entity_test_mul_property_data']));
+    $this->assertFalse(isset($data['revision_table']));
+    $this->assertFalse(isset($data['revision_data_table']));
+  }
+
+  /**
+   * Tests data_table support.
+   */
+  public function testDataTable() {
+    $entity_type = $this->baseEntityType
+      ->set('data_table', 'entity_test_mul_property_data')
+      ->set('id', 'entity_test_mul')
+      ->set('translatable', TRUE)
+      ->setKey('label', 'label');
+
+    $this->setUpEntityType($entity_type);
+
+    // Tests the join definition between the base and the data table.
+    $data = $this->entityTypeManager->getHandler('entity_test_mul', 'views_data')->getViewsData();
+    // TODO: change the base table in the entity type definition to match the
+    // changed entity ID.
+    $base_views_data = $data['entity_test'];
+
+    // Ensure that the base table is set to the data table.
+    $this->assertEquals('id', $data['entity_test_mul_property_data']['table']['base']['field']);
+    $this->assertEquals('Entity test', $data['entity_test_mul_property_data']['table']['base']['title']);
+    $this->assertFalse(isset($data['entity_test']['table']['base']));
+
+    $this->assertEquals('entity_test_mul', $data['entity_test_mul_property_data']['table']['entity type']);
+    $this->assertEquals(FALSE, $data['entity_test_mul_property_data']['table']['entity revision']);
+    $this->assertEquals('Entity test', $data['entity_test_mul_property_data']['table']['group']);
+    $this->assertEquals('entity_test', $data['entity_test']['table']['provider']);
+    $this->assertEquals(['field' => 'label', 'table' => 'entity_test_mul_property_data'], $data['entity_test_mul_property_data']['table']['base']['defaults']);
+
+    // Ensure the join information is set up properly.
+    $this->assertCount(1, $base_views_data['table']['join']);
+    $this->assertEquals(['entity_test_mul_property_data' => ['left_field' => 'id', 'field' => 'id', 'type' => 'INNER']], $base_views_data['table']['join']);
+    $this->assertFalse(isset($data['revision_table']));
+    $this->assertFalse(isset($data['revision_data_table']));
+  }
+
+  /**
+   * Tests revision table without data table support.
+   */
+  public function testRevisionTableWithoutDataTable() {
+    $entity_type = $this->baseEntityType
+      ->set('revision_table', 'entity_test_mulrev_revision')
+      ->set('revision_data_table', NULL)
+      ->set('id', 'entity_test_mulrev')
+      ->setKey('revision', 'revision_id');
+
+    $this->setUpEntityType($entity_type);
+
+    $data = $this->entityTypeManager->getHandler('entity_test_mulrev', 'views_data')->getViewsData();
+
+    $this->assertEquals('Entity test revisions', $data['entity_test_mulrev_revision']['table']['base']['title']);
+    $this->assertEquals('revision_id', $data['entity_test_mulrev_revision']['table']['base']['field']);
+
+    $this->assertEquals(FALSE, $data['entity_test']['table']['entity revision']);
+    $this->assertEquals('entity_test_mulrev', $data['entity_test_mulrev_revision']['table']['entity type']);
+    $this->assertEquals(TRUE, $data['entity_test_mulrev_revision']['table']['entity revision']);
+    $this->assertEquals('entity_test_mulrev', $data['entity_test_mulrev_revision']['table']['entity type']);
+    $this->assertEquals(TRUE, $data['entity_test_mulrev_revision']['table']['entity revision']);
+
+    $this->assertEquals('Entity test revision', $data['entity_test_mulrev_revision']['table']['group']);
+    $this->assertEquals('entity_test', $data['entity_test']['table']['provider']);
+
+    // Ensure the join information is set up properly.
+    // Tests the join definition between the base and the revision table.
+    $revision_data = $data['entity_test_mulrev_revision'];
+    $this->assertCount(1, $revision_data['table']['join']);
+    $this->assertEquals(['entity_test' => ['left_field' => 'revision_id', 'field' => 'revision_id', 'type' => 'INNER']], $revision_data['table']['join']);
+    $this->assertFalse(isset($data['data_table']));
+  }
+
+  /**
+   * Tests revision table with data table support.
+   */
+  public function testRevisionTableWithRevisionDataTableAndDataTable() {
+    $entity_type = $this->baseEntityType
+      ->set('data_table', 'entity_test_mul_property_data')
+      ->set('revision_table', 'entity_test_mulrev_revision')
+      ->set('revision_data_table', 'entity_test_mulrev_property_revision')
+      ->set('id', 'entity_test_mulrev')
+      ->set('translatable', TRUE)
+      ->setKey('revision', 'revision_id');
+    $this->setUpEntityType($entity_type);
+
+    $data = $this->entityTypeManager->getHandler('entity_test_mulrev', 'views_data')->getViewsData();
+
+    $this->assertEquals('Entity test revisions', $data['entity_test_mulrev_property_revision']['table']['base']['title']);
+    $this->assertEquals('revision_id', $data['entity_test_mulrev_property_revision']['table']['base']['field']);
+    $this->assertFalse(isset($data['entity_test_mulrev_revision']['table']['base']));
+
+    $this->assertEquals('entity_test_mulrev', $data['entity_test_mulrev_property_revision']['table']['entity type']);
+    $this->assertEquals('Entity test revision', $data['entity_test_mulrev_revision']['table']['group']);
+    $this->assertEquals('entity_test', $data['entity_test']['table']['provider']);
+
+    // Ensure the join information is set up properly.
+    // Tests the join definition between the base and the revision table.
+    $revision_field_data = $data['entity_test_mulrev_property_revision'];
+    $this->assertCount(1, $revision_field_data['table']['join']);
+    $this->assertEquals([
+      'entity_test_mul_property_data' => [
+        'left_field' => 'revision_id',
+        'field' => 'revision_id',
+        'type' => 'INNER',
+      ],
+    ], $revision_field_data['table']['join']);
+
+    $revision_base_data = $data['entity_test_mulrev_revision'];
+    $this->assertCount(1, $revision_base_data['table']['join']);
+    $this->assertEquals([
+      'entity_test_mulrev_property_revision' => [
+        'left_field' => 'revision_id',
+        'field' => 'revision_id',
+        'type' => 'INNER',
+      ],
+    ], $revision_base_data['table']['join']);
+
+    $this->assertFalse(isset($data['data_table']));
+  }
+
+  /**
+   * Tests revision table with data table support.
+   */
+  public function testRevisionTableWithRevisionDataTable() {
+    $entity_type = $this->baseEntityType
+      ->set('revision_table', 'entity_test_mulrev_revision')
+      ->set('revision_data_table', 'entity_test_mulrev_property_revision')
+      ->set('id', 'entity_test_mulrev')
+      ->set('translatable', TRUE)
+      ->setKey('revision', 'revision_id');
+    $this->setUpEntityType($entity_type);
+
+    $data = $this->entityTypeManager->getHandler('entity_test_mulrev', 'views_data')->getViewsData();
+
+    $this->assertEquals('Entity test revisions', $data['entity_test_mulrev_property_revision']['table']['base']['title']);
+    $this->assertEquals('revision_id', $data['entity_test_mulrev_property_revision']['table']['base']['field']);
+    $this->assertFalse(isset($data['entity_test_mulrev_revision']['table']['base']));
+
+    $this->assertEquals('entity_test_mulrev', $data['entity_test_mulrev_property_revision']['table']['entity type']);
+    $this->assertEquals('Entity test revision', $data['entity_test_mulrev_revision']['table']['group']);
+    $this->assertEquals('entity_test', $data['entity_test']['table']['provider']);
+
+    // Ensure the join information is set up properly.
+    // Tests the join definition between the base and the revision table.
+    $revision_field_data = $data['entity_test_mulrev_property_revision'];
+    $this->assertCount(1, $revision_field_data['table']['join']);
+    $this->assertEquals([
+      'entity_test_mulrev_field_data' => [
+        'left_field' => 'revision_id',
+        'field' => 'revision_id',
+        'type' => 'INNER',
+      ],
+    ], $revision_field_data['table']['join']);
+
+    $revision_base_data = $data['entity_test_mulrev_revision'];
+    $this->assertCount(1, $revision_base_data['table']['join']);
+    $this->assertEquals([
+      'entity_test_mulrev_property_revision' => [
+        'left_field' => 'revision_id',
+        'field' => 'revision_id',
+        'type' => 'INNER',
+      ],
+    ], $revision_base_data['table']['join']);
+    $this->assertFalse(isset($data['data_table']));
+  }
+
+  /**
+   * Tests fields on the base table.
+   */
+  public function testBaseTableFields() {
+    $data = $this->entityTypeManager->getHandler('entity_test', 'views_data')->getViewsData();
+
+    $this->assertNumericField($data['entity_test']['id']);
+    $this->assertViewsDataField($data['entity_test']['id'], 'id');
+    $this->assertUuidField($data['entity_test']['uuid']);
+    $this->assertViewsDataField($data['entity_test']['uuid'], 'uuid');
+    $this->assertStringField($data['entity_test']['type']);
+    $this->assertEquals('type', $data['entity_test']['type']['entity field']);
+
+    $this->assertLanguageField($data['entity_test']['langcode']);
+    $this->assertViewsDataField($data['entity_test']['langcode'], 'langcode');
+    $this->assertEquals('Original language', $data['entity_test']['langcode']['title']);
+
+    $this->assertStringField($data['entity_test']['name']);
+    $this->assertViewsDataField($data['entity_test']['name'], 'name');
+
+    $this->assertLongTextField($data['entity_test'], 'description');
+    $this->assertViewsDataField($data['entity_test']['description__value'], 'description');
+    $this->assertViewsDataField($data['entity_test']['description__format'], 'description');
+
+    $this->assertUriField($data['entity_test']['homepage']);
+    $this->assertViewsDataField($data['entity_test']['homepage'], 'homepage');
+
+    $this->assertEntityReferenceField($data['entity_test']['user_id']);
+    $this->assertViewsDataField($data['entity_test']['user_id'], 'user_id');
+
+    $relationship = $data['entity_test']['user_id']['relationship'];
+    $this->assertEquals('users_field_data', $relationship['base']);
+    $this->assertEquals('uid', $relationship['base field']);
+
+    // The string field name should be used as the 'entity field' but the actual
+    // field should reflect what the column mapping is using for multi-value
+    // base fields NOT just the field name. The actual column name returned from
+    // mappings in the test mocks is 'value'.
+    $this->assertStringField($data['entity_test__string']['string_value']);
+    $this->assertViewsDataField($data['entity_test__string']['string_value'], 'string');
+    $this->assertEquals([
+      'left_field' => 'id',
+      'field' => 'entity_id',
+      'extra' => [[
+          'field' => 'deleted',
+          'value' => 0,
+          'numeric' => TRUE,
+        ],
+      ],
+    ], $data['entity_test__string']['table']['join']['entity_test']);
+  }
+
+  /**
+   * Tests fields on the data table.
+   */
+  public function testDataTableFields() {
+    $entity_test_type = new ConfigEntityType([
+      'class' => ConfigEntityBase::class,
+      'id' => 'entity_test_bundle',
+      'entity_keys' => [
+        'id' => 'type',
+        'label' => 'name',
+      ],
+    ]);
+    $this->setUpEntityType($entity_test_type);
+
+    $entity_type = $this->baseEntityType
+      ->set('data_table', 'entity_test_mul_property_data')
+      ->set('base_table', 'entity_test_mul')
+      ->set('translatable', TRUE)
+      ->set('id', 'entity_test_mul')
+      ->set('bundle_entity_type', 'entity_test_bundle')
+      ->setKey('bundle', 'type');
+
+    $base_field_definitions = $this->commonBaseFields;
+    $base_field_definitions['type'] = BaseFieldDefinition::create('entity_reference')
+      ->setLabel('entity test type')
+      ->setSetting('target_type', 'entity_test_bundle');
+
+    $this->setUpEntityType($entity_type, $base_field_definitions);
+
+    $data = $this->entityTypeManager->getHandler('entity_test_mul', 'views_data')->getViewsData();
+
+    // Check the base fields.
+    $this->assertFalse(isset($data['entity_test_mul']['id']));
+    $this->assertFalse(isset($data['entity_test_mul']['type']));
+    $this->assertUuidField($data['entity_test_mul']['uuid']);
+    $this->assertViewsDataField($data['entity_test_mul']['uuid'], 'uuid');
+
+    $this->assertFalse(isset($data['entity_test_mul']['type']['relationship']));
+
+    // Also ensure that field_data only fields don't appear on the base table.
+    $this->assertFalse(isset($data['entity_test_mul']['name']));
+    $this->assertFalse(isset($data['entity_test_mul']['description']));
+    $this->assertFalse(isset($data['entity_test_mul']['description__value']));
+    $this->assertFalse(isset($data['entity_test_mul']['description__format']));
+    $this->assertFalse(isset($data['entity_test_mul']['user_id']));
+    $this->assertFalse(isset($data['entity_test_mul']['homepage']));
+
+    // Check the data fields.
+    $this->assertNumericField($data['entity_test_mul_property_data']['id']);
+    $this->assertViewsDataField($data['entity_test_mul_property_data']['id'], 'id');
+
+    $this->assertBundleField($data['entity_test_mul_property_data']['type']);
+    $this->assertViewsDataField($data['entity_test_mul_property_data']['type'], 'type');
+
+    $this->assertLanguageField($data['entity_test_mul_property_data']['langcode']);
+    $this->assertViewsDataField($data['entity_test_mul_property_data']['langcode'], 'langcode');
+    $this->assertEquals('Translation language', $data['entity_test_mul_property_data']['langcode']['title']);
+
+    $this->assertStringField($data['entity_test_mul_property_data']['name']);
+    $this->assertViewsDataField($data['entity_test_mul_property_data']['name'], 'name');
+
+    $this->assertLongTextField($data['entity_test_mul_property_data'], 'description');
+    $this->assertViewsDataField($data['entity_test_mul_property_data']['description__value'], 'description');
+    $this->assertViewsDataField($data['entity_test_mul_property_data']['description__format'], 'description');
+
+    $this->assertUriField($data['entity_test_mul_property_data']['homepage']);
+    $this->assertViewsDataField($data['entity_test_mul_property_data']['homepage'], 'homepage');
+
+    $this->assertEntityReferenceField($data['entity_test_mul_property_data']['user_id']);
+    $this->assertViewsDataField($data['entity_test_mul_property_data']['user_id'], 'user_id');
+    $relationship = $data['entity_test_mul_property_data']['user_id']['relationship'];
+    $this->assertEquals('users_field_data', $relationship['base']);
+    $this->assertEquals('uid', $relationship['base field']);
+
+    $this->assertStringField($data['entity_test_mul__string']['string_value']);
+    $this->assertViewsDataField($data['entity_test_mul__string']['string_value'], 'string');
+    $this->assertEquals([
+      'left_field' => 'id',
+      'field' => 'entity_id',
+      'extra' => [[
+          'field' => 'deleted',
+          'value' => 0,
+          'numeric' => TRUE,
+        ],
+      ],
+    ], $data['entity_test_mul__string']['table']['join']['entity_test_mul_property_data']);
+  }
+
+  /**
+   * Tests fields on the revision table.
+   */
+  public function testRevisionTableFields() {
+    $entity_type = $this->baseEntityType
+      ->set('id', 'entity_test_mulrev')
+      ->set('base_table', 'entity_test_mulrev')
+      ->set('revision_table', 'entity_test_mulrev_revision')
+      ->set('data_table', 'entity_test_mulrev_property_data')
+      ->set('revision_data_table', 'entity_test_mulrev_property_revision')
+      ->set('translatable', TRUE);
+
+    $base_field_definitions = $this->commonBaseFields;
+
+    $base_field_definitions['name']->setRevisionable(TRUE);
+    $base_field_definitions['description']->setRevisionable(TRUE);
+    $base_field_definitions['homepage']->setRevisionable(TRUE);
+    $base_field_definitions['user_id']->setRevisionable(TRUE);
+
+    $base_field_definitions['non_rev_field'] = BaseFieldDefinition::create('string')
+      ->setLabel(t('Non Revisionable Field'))
+      ->setDescription(t('A non-revisionable test field.'))
+      ->setRevisionable(FALSE)
+      ->setTranslatable(TRUE)
+      ->setCardinality(1)
+      ->setReadOnly(TRUE);
+
+    $base_field_definitions['non_mul_field'] = BaseFieldDefinition::create('string')
+      ->setLabel(t('Non translatable'))
+      ->setDescription(t('A non-translatable string field'))
+      ->setRevisionable(TRUE);
+
+    $this->setUpEntityType($entity_type, $base_field_definitions);
+
+    $data = $this->entityTypeManager->getHandler('entity_test_mulrev', 'views_data')->getViewsData();
+
+    // Check the base fields.
+    $this->assertFalse(isset($data['entity_test_mulrev']['id']));
+    $this->assertFalse(isset($data['entity_test_mulrev']['type']));
+    $this->assertFalse(isset($data['entity_test_mulrev']['revision_id']));
+    $this->assertUuidField($data['entity_test_mulrev']['uuid']);
+    $this->assertViewsDataField($data['entity_test_mulrev']['uuid'], 'uuid');
+
+    // Also ensure that field_data only fields don't appear on the base table.
+    $this->assertFalse(isset($data['entity_test_mulrev']['name']));
+    $this->assertFalse(isset($data['entity_test_mul']['description']));
+    $this->assertFalse(isset($data['entity_test_mul']['description__value']));
+    $this->assertFalse(isset($data['entity_test_mul']['description__format']));
+    $this->assertFalse(isset($data['entity_test_mul']['homepage']));
+    // $this->assertFalse(isset($data['entity_test_mulrev']['langcode']));
+    $this->assertFalse(isset($data['entity_test_mulrev']['user_id']));
+
+    // Check the revision fields. The revision ID should only appear in the data
+    // table.
+    $this->assertFalse(isset($data['entity_test_mulrev_revision']['revision_id']));
+
+    // Also ensure that field_data only fields don't appear on the revision table.
+    $this->assertFalse(isset($data['entity_test_mulrev_revision']['id']));
+    $this->assertFalse(isset($data['entity_test_mulrev_revision']['name']));
+    $this->assertFalse(isset($data['entity_test_mulrev_revision']['description']));
+    $this->assertFalse(isset($data['entity_test_mulrev_revision']['description__value']));
+    $this->assertFalse(isset($data['entity_test_mulrev_revision']['description__format']));
+    $this->assertFalse(isset($data['entity_test_mulrev_revision']['homepage']));
+    $this->assertFalse(isset($data['entity_test_mulrev_revision']['user_id']));
+
+    // Check the data fields.
+    $this->assertNumericField($data['entity_test_mulrev_property_data']['id']);
+    $this->assertViewsDataField($data['entity_test_mulrev_property_data']['id'], 'id');
+    $this->assertNumericField($data['entity_test_mulrev_property_data']['revision_id']);
+    $this->assertViewsDataField($data['entity_test_mulrev_property_data']['revision_id'], 'revision_id');
+    $this->assertLanguageField($data['entity_test_mulrev_property_data']['langcode']);
+    $this->assertViewsDataField($data['entity_test_mulrev_property_data']['langcode'], 'langcode');
+    $this->assertStringField($data['entity_test_mulrev_property_data']['name']);
+    $this->assertViewsDataField($data['entity_test_mulrev_property_data']['name'], 'name');
+
+    $this->assertLongTextField($data['entity_test_mulrev_property_data'], 'description');
+    $this->assertViewsDataField($data['entity_test_mulrev_property_data']['description__value'], 'description');
+    $this->assertViewsDataField($data['entity_test_mulrev_property_data']['description__format'], 'description');
+    $this->assertUriField($data['entity_test_mulrev_property_data']['homepage']);
+    $this->assertViewsDataField($data['entity_test_mulrev_property_data']['homepage'], 'homepage');
+
+    $this->assertEntityReferenceField($data['entity_test_mulrev_property_data']['user_id']);
+    $this->assertViewsDataField($data['entity_test_mulrev_property_data']['user_id'], 'user_id');
+    $relationship = $data['entity_test_mulrev_property_data']['user_id']['relationship'];
+    $this->assertEquals('users_field_data', $relationship['base']);
+    $this->assertEquals('uid', $relationship['base field']);
+
+    // Check the property data fields.
+    $this->assertNumericField($data['entity_test_mulrev_property_revision']['id']);
+    $this->assertViewsDataField($data['entity_test_mulrev_property_revision']['id'], 'id');
+
+    $this->assertLanguageField($data['entity_test_mulrev_property_revision']['langcode']);
+    $this->assertViewsDataField($data['entity_test_mulrev_property_revision']['langcode'], 'langcode');
+    $this->assertEquals('Translation language', $data['entity_test_mulrev_property_revision']['langcode']['title']);
+
+    $this->assertStringField($data['entity_test_mulrev_property_revision']['name']);
+    $this->assertViewsDataField($data['entity_test_mulrev_property_revision']['name'], 'name');
+
+    $this->assertLongTextField($data['entity_test_mulrev_property_revision'], 'description');
+    $this->assertViewsDataField($data['entity_test_mulrev_property_revision']['description__value'], 'description');
+    $this->assertViewsDataField($data['entity_test_mulrev_property_revision']['description__format'], 'description');
+
+    $this->assertUriField($data['entity_test_mulrev_property_revision']['homepage']);
+    $this->assertViewsDataField($data['entity_test_mulrev_property_revision']['homepage'], 'homepage');
+
+    $this->assertEntityReferenceField($data['entity_test_mulrev_property_revision']['user_id']);
+    $this->assertViewsDataField($data['entity_test_mulrev_property_revision']['user_id'], 'user_id');
+    $relationship = $data['entity_test_mulrev_property_revision']['user_id']['relationship'];
+    $this->assertEquals('users_field_data', $relationship['base']);
+    $this->assertEquals('uid', $relationship['base field']);
+
+    $this->assertStringField($data['entity_test_mulrev__string']['string_value']);
+    $this->assertViewsDataField($data['entity_test_mulrev__string']['string_value'], 'string');
+    $this->assertEquals([
+      'left_field' => 'id',
+      'field' => 'entity_id',
+      'extra' => [[
+          'field' => 'deleted',
+          'value' => 0,
+          'numeric' => TRUE,
+        ],
+      ],
+    ], $data['entity_test_mulrev__string']['table']['join']['entity_test_mulrev_property_data']);
+
+    $this->assertStringField($data['entity_test_mulrev_revision__string']['string_value']);
+    $this->assertViewsDataField($data['entity_test_mulrev_revision__string']['string_value'], 'string');
+    $this->assertEquals([
+      'left_field' => 'revision_id',
+      'field' => 'entity_id',
+      'extra' => [[
+          'field' => 'deleted',
+          'value' => 0,
+          'numeric' => TRUE,
+        ],
+      ],
+    ], $data['entity_test_mulrev_revision__string']['table']['join']['entity_test_mulrev_property_revision']);
+  }
+
+  /**
+   * Tests generic stuff per field.
+   *
+   * @param array $data
+   *   The views data to check.
+   * @param string $field_name
+   *   The entity field name.
+   */
+  protected function assertViewsDataField($data, $field_name) {
+    $this->assertEquals($field_name, $data['entity field']);
+  }
+
+  /**
+   * Tests views data for a string field.
+   *
+   * @param $data
+   *   The views data to check.
+   */
+  protected function assertStringField($data) {
+    $this->assertEquals('field', $data['field']['id']);
+    $this->assertEquals('string', $data['filter']['id']);
+    $this->assertEquals('string', $data['argument']['id']);
+    $this->assertEquals('standard', $data['sort']['id']);
+  }
+
+  /**
+   * Tests views data for a URI field.
+   *
+   * @param $data
+   *   The views data to check.
+   */
+  protected function assertUriField($data) {
+    $this->assertEquals('field', $data['field']['id']);
+    $this->assertEquals('string', $data['field']['default_formatter']);
+    $this->assertEquals('string', $data['filter']['id']);
+    $this->assertEquals('string', $data['argument']['id']);
+    $this->assertEquals('standard', $data['sort']['id']);
+  }
+
+  /**
+   * Tests views data for a long text field.
+   *
+   * @param $data
+   *   The views data for the table this field is in.
+   * @param $field_name
+   *   The name of the field being checked.
+   */
+  protected function assertLongTextField($data, $field_name) {
+    $value_field = $data[$field_name . '__value'];
+    $this->assertEquals('field', $value_field['field']['id']);
+    $this->assertEquals($field_name . '__format', $value_field['field']['format']);
+    $this->assertEquals('string', $value_field['filter']['id']);
+    $this->assertEquals('string', $value_field['argument']['id']);
+    $this->assertEquals('standard', $value_field['sort']['id']);
+
+    $this->assertStringField($data[$field_name . '__format']);
+  }
+
+  /**
+   * Tests views data for a UUID field.
+   *
+   * @param array $data
+   *   The views data to check.
+   */
+  protected function assertUuidField($data) {
+    // @todo Can we provide additional support for UUIDs in views?
+    $this->assertEquals('field', $data['field']['id']);
+    $this->assertFalse($data['field']['click sortable']);
+    $this->assertEquals('string', $data['filter']['id']);
+    $this->assertEquals('string', $data['argument']['id']);
+    $this->assertEquals('standard', $data['sort']['id']);
+  }
+
+  /**
+   * Tests views data for a numeric field.
+   *
+   * @param array $data
+   *   The views data to check.
+   */
+  protected function assertNumericField($data) {
+    $this->assertEquals('field', $data['field']['id']);
+    $this->assertEquals('numeric', $data['filter']['id']);
+    $this->assertEquals('numeric', $data['argument']['id']);
+    $this->assertEquals('standard', $data['sort']['id']);
+  }
+
+  /**
+   * Tests views data for a language field.
+   *
+   * @param array $data
+   *   The views data to check.
+   */
+  protected function assertLanguageField($data) {
+    $this->assertEquals('field', $data['field']['id']);
+    $this->assertEquals('language', $data['filter']['id']);
+    $this->assertEquals('language', $data['argument']['id']);
+    $this->assertEquals('standard', $data['sort']['id']);
+  }
+
+  /**
+   * Tests views data for an entity reference field.
+   */
+  protected function assertEntityReferenceField($data) {
+    $this->assertEquals('field', $data['field']['id']);
+    $this->assertEquals('numeric', $data['filter']['id']);
+    $this->assertEquals('numeric', $data['argument']['id']);
+    $this->assertEquals('standard', $data['sort']['id']);
+  }
+
+  /**
+   * Tests views data for a bundle field.
+   */
+  protected function assertBundleField($data) {
+    $this->assertEquals('field', $data['field']['id']);
+    $this->assertEquals('bundle', $data['filter']['id']);
+    $this->assertEquals('string', $data['argument']['id']);
+    $this->assertEquals('standard', $data['sort']['id']);
+  }
+
+}
+
+/**
+ * Entity type class which allows changing the entity keys.
+ */
+class TestEntityType extends ContentEntityType {
+
+  /**
+   * Sets a specific entity key.
+   *
+   * @param string $key
+   *   The name of the entity key.
+   * @param string $value
+   *   The new value of the key.
+   *
+   * @return $this
+   */
+  public function setKey($key, $value) {
+    $this->entity_keys[$key] = $value;
+    return $this;
+  }
+
+}
+
+/**
+ * Generic entity class for our test entity types.
+ *
+ * Allows mocked base field definitions.
+ */
+class ViewsTestEntity extends ContentEntityBase {
+
+  /**
+   * The mocked base fields for test entity types.
+   *
+   * An array keyed by entity type ID, whose values are arrays of base field
+   * definitions.
+   *
+   * @var array
+   */
+  protected static $mockedBaseFieldDefinitions = [];
+
+  /**
+   * Sets up the mocked base field definitions.
+   *
+   * @param string $entity_type_id
+   *   The entity type ID.
+   * @param array $definitions
+   *   The array of base field definitions to mock. These are added to the
+   *   defaults ones from the parent class.
+   */
+  public static function setMockedBaseFieldDefinitions(string $entity_type_id, array $definitions) {
+    static::$mockedBaseFieldDefinitions[$entity_type_id] = $definitions;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public static function baseFieldDefinitions(EntityTypeInterface $entity_type) {
+    $fields = parent::baseFieldDefinitions($entity_type);
+
+    if (isset(static::$mockedBaseFieldDefinitions[$entity_type->id()])) {
+      $mocked_fields = static::$mockedBaseFieldDefinitions[$entity_type->id()];
+      // Mocked fields take priority over ones from the base class.
+      $fields = $mocked_fields + $fields;
+    }
+
+    return $fields;
+  }
+
+}
diff --git a/web/core/modules/views/tests/src/Unit/EntityViewsDataTest.php b/web/core/modules/views/tests/src/Unit/EntityViewsDataTest.php
deleted file mode 100644
index 6e0621e40fde5b6d0b9b87e9c4343ce4360b0576..0000000000000000000000000000000000000000
--- a/web/core/modules/views/tests/src/Unit/EntityViewsDataTest.php
+++ /dev/null
@@ -1,1197 +0,0 @@
-<?php
-
-/**
- * @file
- * Contains \Drupal\Tests\views\Unit\EntityViewsDataTest.
- */
-
-namespace Drupal\Tests\views\Unit;
-
-use Drupal\Core\Config\Entity\ConfigEntityType;
-use Drupal\Core\Entity\ContentEntityType;
-use Drupal\Core\Entity\EntityFieldManagerInterface;
-use Drupal\Core\Entity\EntityTypeInterface;
-use Drupal\Core\Entity\EntityTypeManagerInterface;
-use Drupal\Core\Entity\Sql\DefaultTableMapping;
-use Drupal\Core\Field\BaseFieldDefinition;
-use Drupal\Core\Field\Plugin\Field\FieldType\EntityReferenceItem;
-use Drupal\Core\Field\Plugin\Field\FieldType\IntegerItem;
-use Drupal\Core\Field\Plugin\Field\FieldType\LanguageItem;
-use Drupal\Core\Field\Plugin\Field\FieldType\StringItem;
-use Drupal\Core\Field\Plugin\Field\FieldType\UriItem;
-use Drupal\Core\Field\Plugin\Field\FieldType\UuidItem;
-use Drupal\Core\State\StateInterface;
-use Drupal\Core\TypedData\TypedDataManagerInterface;
-use Drupal\text\Plugin\Field\FieldType\TextLongItem;
-use Drupal\entity_test\Entity\EntityTest;
-use Drupal\entity_test\Entity\EntityTestMul;
-use Drupal\entity_test\Entity\EntityTestMulRev;
-use Drupal\Tests\UnitTestCase;
-use Drupal\views\EntityViewsData;
-use Prophecy\Argument;
-use Symfony\Component\DependencyInjection\ContainerBuilder;
-
-/**
- * @coversDefaultClass \Drupal\views\EntityViewsData
- * @group Views
- */
-class EntityViewsDataTest extends UnitTestCase {
-
-  /**
-   * Entity info to use in this test.
-   *
-   * @var \Drupal\Core\Entity\EntityTypeInterface|\Drupal\Tests\views\Unit\TestEntityType
-   */
-  protected $baseEntityType;
-
-  /**
-   * The mocked entity storage.
-   *
-   * @var \Drupal\Core\Entity\Sql\SqlContentEntityStorage|\PHPUnit\Framework\MockObject\MockObject
-   */
-  protected $entityStorage;
-
-  /**
-   * The mocked entity field manager.
-   *
-   * @var \Drupal\Core\Entity\EntityFieldManagerInterface|\PHPUnit\Framework\MockObject\MockObject
-   */
-  protected $entityFieldManager;
-
-
-  /**
-   * The mocked entity type manager.
-   *
-   * @var \Drupal\Core\Entity\EntityTypeManagerInterface|\PHPUnit\Framework\MockObject\MockObject
-   */
-  protected $entityTypeManager;
-
-  /**
-   * The mocked module handler.
-   *
-   * @var \Drupal\Core\Extension\ModuleHandlerInterface|\PHPUnit\Framework\MockObject\MockObject
-   */
-  protected $moduleHandler;
-
-  /**
-   * The mocked translation manager.
-   *
-   * @var \Drupal\Core\StringTranslation\TranslationInterface|\PHPUnit\Framework\MockObject\MockObject
-   */
-  protected $translationManager;
-
-  /**
-   * The tested entity views controller.
-   *
-   * @var \Drupal\Tests\views\Unit\TestEntityViewsData
-   */
-  protected $viewsData;
-
-  /**
-   * {@inheritdoc}
-   */
-  protected function setUp(): void {
-    $this->entityStorage = $this->getMockBuilder('Drupal\Core\Entity\Sql\SqlContentEntityStorage')
-      ->disableOriginalConstructor()
-      ->getMock();
-    $this->entityTypeManager = $this->createMock(EntityTypeManagerInterface::class);
-    $this->entityFieldManager = $this->createMock(EntityFieldManagerInterface::class);
-
-    $typed_data_manager = $this->createMock(TypedDataManagerInterface::class);
-    $typed_data_manager->expects($this->any())
-      ->method('createDataDefinition')
-      ->willReturn($this->createMock('Drupal\Core\TypedData\DataDefinitionInterface'));
-
-    $typed_data_manager->expects($this->any())
-      ->method('getDefinition')
-      ->will($this->returnValueMap([
-        'entity:user' => ['class' => '\Drupal\Core\TypedData\DataDefinitionInterface'],
-        'field_item:string_long' => ['class' => '\Drupal\Core\Field\Plugin\Field\FieldType\StringLongItem'],
-      ]));
-
-    $this->baseEntityType = new TestEntityType([
-      'base_table' => 'entity_test',
-      'id' => 'entity_test',
-      'label' => 'Entity test',
-      'entity_keys' => [
-        'uuid' => 'uuid',
-        'id' => 'id',
-        'langcode' => 'langcode',
-        'bundle' => 'type',
-        'revision' => 'revision_id',
-      ],
-      'provider' => 'entity_test',
-      'list_cache_contexts' => ['entity_test_list_cache_context'],
-    ]);
-
-    $this->translationManager = $this->getStringTranslationStub();
-    $this->baseEntityType->setStringTranslation($this->translationManager);
-    $this->moduleHandler = $this->createMock('Drupal\Core\Extension\ModuleHandlerInterface');
-
-    $this->viewsData = new TestEntityViewsData($this->baseEntityType, $this->entityStorage, $this->entityTypeManager, $this->moduleHandler, $this->translationManager, $this->entityFieldManager);
-
-    $field_type_manager = $this->getMockBuilder('Drupal\Core\Field\FieldTypePluginManager')
-      ->disableOriginalConstructor()
-      ->getMock();
-    $field_type_manager->expects($this->any())
-      ->method('getDefaultStorageSettings')
-      ->willReturn([]);
-    $field_type_manager->expects($this->any())
-      ->method('getDefaultFieldSettings')
-      ->willReturn([]);
-
-    $state = $this->prophesize(StateInterface::class);
-    $state->get(Argument::any(), [])->willReturn([]);
-
-    $container = new ContainerBuilder();
-    $container->set('plugin.manager.field.field_type', $field_type_manager);
-    $container->set('entity_type.manager', $this->entityTypeManager);
-    $container->set('entity_field.manager', $this->entityFieldManager);
-    $container->set('typed_data_manager', $typed_data_manager);
-    $container->set('state', $state->reveal());
-    \Drupal::setContainer($container);
-  }
-
-  /**
-   * Helper method to setup base fields.
-   *
-   * @param \Drupal\Core\Field\BaseFieldDefinition[] $base_fields
-   *   The base fields which are adapted.
-   *
-   * @return \Drupal\Core\Field\BaseFieldDefinition[]
-   *   The setup base fields.
-   */
-  protected function setupBaseFields(array $base_fields) {
-    // Add a description field to the fields supplied by the EntityTest
-    // classes. This example comes from the taxonomy Term entity.
-    $base_fields['description'] = BaseFieldDefinition::create('text_long')
-      ->setLabel('Description')
-      ->setDescription('A description of the term.')
-      ->setTranslatable(TRUE)
-      ->setDisplayOptions('view', [
-          'label' => 'hidden',
-          'type' => 'text_default',
-          'weight' => 0,
-        ])
-      ->setDisplayConfigurable('view', TRUE)
-      ->setDisplayOptions('form', [
-          'type' => 'text_textfield',
-          'weight' => 0,
-        ])
-      ->setDisplayConfigurable('form', TRUE);
-
-    // Add a URL field; this example is from the Comment entity.
-    $base_fields['homepage'] = BaseFieldDefinition::create('uri')
-      ->setLabel('Homepage')
-      ->setDescription("The comment author's home page address.")
-      ->setTranslatable(TRUE)
-      ->setSetting('max_length', 255);
-
-    // A base field with cardinality > 1
-    $base_fields['string'] = BaseFieldDefinition::create('string')
-      ->setLabel('Strong')
-      ->setTranslatable(TRUE)
-      ->setCardinality(2);
-
-    foreach ($base_fields as $name => $base_field) {
-      $base_field->setName($name);
-    }
-    return $base_fields;
-  }
-
-  /**
-   * Tests base tables.
-   */
-  public function testBaseTables() {
-    $data = $this->viewsData->getViewsData();
-
-    $this->assertEquals('entity_test', $data['entity_test']['table']['entity type']);
-    $this->assertEquals(FALSE, $data['entity_test']['table']['entity revision']);
-    $this->assertEquals('Entity test', $data['entity_test']['table']['group']);
-    $this->assertEquals('entity_test', $data['entity_test']['table']['provider']);
-
-    $this->assertEquals('id', $data['entity_test']['table']['base']['field']);
-    $this->assertEquals(['entity_test_list_cache_context'], $data['entity_test']['table']['base']['cache_contexts']);
-    $this->assertEquals('Entity test', $data['entity_test']['table']['base']['title']);
-
-    $this->assertFalse(isset($data['entity_test']['table']['defaults']));
-
-    $this->assertFalse(isset($data['entity_test_mul_property_data']));
-    $this->assertFalse(isset($data['revision_table']));
-    $this->assertFalse(isset($data['revision_data_table']));
-  }
-
-  /**
-   * Tests data_table support.
-   */
-  public function testDataTable() {
-    $entity_type = $this->baseEntityType
-      ->set('data_table', 'entity_test_mul_property_data')
-      ->set('id', 'entity_test_mul')
-      ->set('translatable', TRUE)
-      ->setKey('label', 'label');
-
-    $this->viewsData->setEntityType($entity_type);
-
-    // Tests the join definition between the base and the data table.
-    $data = $this->viewsData->getViewsData();
-    $base_views_data = $data['entity_test'];
-
-    // Ensure that the base table is set to the data table.
-    $this->assertEquals('id', $data['entity_test_mul_property_data']['table']['base']['field']);
-    $this->assertEquals('Entity test', $data['entity_test_mul_property_data']['table']['base']['title']);
-    $this->assertFalse(isset($data['entity_test']['table']['base']));
-
-    $this->assertEquals('entity_test_mul', $data['entity_test_mul_property_data']['table']['entity type']);
-    $this->assertEquals(FALSE, $data['entity_test_mul_property_data']['table']['entity revision']);
-    $this->assertEquals('Entity test', $data['entity_test_mul_property_data']['table']['group']);
-    $this->assertEquals('entity_test', $data['entity_test']['table']['provider']);
-    $this->assertEquals(['field' => 'label', 'table' => 'entity_test_mul_property_data'], $data['entity_test_mul_property_data']['table']['base']['defaults']);
-
-    // Ensure the join information is set up properly.
-    $this->assertCount(1, $base_views_data['table']['join']);
-    $this->assertEquals(['entity_test_mul_property_data' => ['left_field' => 'id', 'field' => 'id', 'type' => 'INNER']], $base_views_data['table']['join']);
-    $this->assertFalse(isset($data['revision_table']));
-    $this->assertFalse(isset($data['revision_data_table']));
-  }
-
-  /**
-   * Tests revision table without data table support.
-   */
-  public function testRevisionTableWithoutDataTable() {
-    $entity_type = $this->baseEntityType
-      ->set('revision_table', 'entity_test_mulrev_revision')
-      ->set('revision_data_table', NULL)
-      ->set('id', 'entity_test_mulrev')
-      ->setKey('revision', 'revision_id');
-    $this->viewsData->setEntityType($entity_type);
-
-    $data = $this->viewsData->getViewsData();
-
-    $this->assertEquals('Entity test revisions', $data['entity_test_mulrev_revision']['table']['base']['title']);
-    $this->assertEquals('revision_id', $data['entity_test_mulrev_revision']['table']['base']['field']);
-
-    $this->assertEquals(FALSE, $data['entity_test']['table']['entity revision']);
-    $this->assertEquals('entity_test_mulrev', $data['entity_test_mulrev_revision']['table']['entity type']);
-    $this->assertEquals(TRUE, $data['entity_test_mulrev_revision']['table']['entity revision']);
-    $this->assertEquals('entity_test_mulrev', $data['entity_test_mulrev_revision']['table']['entity type']);
-    $this->assertEquals(TRUE, $data['entity_test_mulrev_revision']['table']['entity revision']);
-
-    $this->assertEquals('Entity test revision', $data['entity_test_mulrev_revision']['table']['group']);
-    $this->assertEquals('entity_test', $data['entity_test']['table']['provider']);
-
-    // Ensure the join information is set up properly.
-    // Tests the join definition between the base and the revision table.
-    $revision_data = $data['entity_test_mulrev_revision'];
-    $this->assertCount(1, $revision_data['table']['join']);
-    $this->assertEquals(['entity_test' => ['left_field' => 'revision_id', 'field' => 'revision_id', 'type' => 'INNER']], $revision_data['table']['join']);
-    $this->assertFalse(isset($data['data_table']));
-  }
-
-  /**
-   * Tests revision table with data table support.
-   */
-  public function testRevisionTableWithRevisionDataTableAndDataTable() {
-    $entity_type = $this->baseEntityType
-      ->set('data_table', 'entity_test_mul_property_data')
-      ->set('revision_table', 'entity_test_mulrev_revision')
-      ->set('revision_data_table', 'entity_test_mulrev_property_revision')
-      ->set('id', 'entity_test_mulrev')
-      ->set('translatable', TRUE)
-      ->setKey('revision', 'revision_id');
-    $this->viewsData->setEntityType($entity_type);
-
-    $data = $this->viewsData->getViewsData();
-
-    $this->assertEquals('Entity test revisions', $data['entity_test_mulrev_property_revision']['table']['base']['title']);
-    $this->assertEquals('revision_id', $data['entity_test_mulrev_property_revision']['table']['base']['field']);
-    $this->assertFalse(isset($data['entity_test_mulrev_revision']['table']['base']));
-
-    $this->assertEquals('entity_test_mulrev', $data['entity_test_mulrev_property_revision']['table']['entity type']);
-    $this->assertEquals('Entity test revision', $data['entity_test_mulrev_revision']['table']['group']);
-    $this->assertEquals('entity_test', $data['entity_test']['table']['provider']);
-
-    // Ensure the join information is set up properly.
-    // Tests the join definition between the base and the revision table.
-    $revision_field_data = $data['entity_test_mulrev_property_revision'];
-    $this->assertCount(1, $revision_field_data['table']['join']);
-    $this->assertEquals([
-      'entity_test_mul_property_data' => [
-        'left_field' => 'revision_id',
-        'field' => 'revision_id',
-        'type' => 'INNER',
-      ],
-    ], $revision_field_data['table']['join']);
-
-    $revision_base_data = $data['entity_test_mulrev_revision'];
-    $this->assertCount(1, $revision_base_data['table']['join']);
-    $this->assertEquals([
-      'entity_test_mulrev_property_revision' => [
-        'left_field' => 'revision_id',
-        'field' => 'revision_id',
-        'type' => 'INNER',
-      ],
-    ], $revision_base_data['table']['join']);
-
-    $this->assertFalse(isset($data['data_table']));
-  }
-
-  /**
-   * Tests revision table with data table support.
-   */
-  public function testRevisionTableWithRevisionDataTable() {
-    $entity_type = $this->baseEntityType
-      ->set('revision_table', 'entity_test_mulrev_revision')
-      ->set('revision_data_table', 'entity_test_mulrev_property_revision')
-      ->set('id', 'entity_test_mulrev')
-      ->set('translatable', TRUE)
-      ->setKey('revision', 'revision_id');
-    $this->viewsData->setEntityType($entity_type);
-
-    $data = $this->viewsData->getViewsData();
-
-    $this->assertEquals('Entity test revisions', $data['entity_test_mulrev_property_revision']['table']['base']['title']);
-    $this->assertEquals('revision_id', $data['entity_test_mulrev_property_revision']['table']['base']['field']);
-    $this->assertFalse(isset($data['entity_test_mulrev_revision']['table']['base']));
-
-    $this->assertEquals('entity_test_mulrev', $data['entity_test_mulrev_property_revision']['table']['entity type']);
-    $this->assertEquals('Entity test revision', $data['entity_test_mulrev_revision']['table']['group']);
-    $this->assertEquals('entity_test', $data['entity_test']['table']['provider']);
-
-    // Ensure the join information is set up properly.
-    // Tests the join definition between the base and the revision table.
-    $revision_field_data = $data['entity_test_mulrev_property_revision'];
-    $this->assertCount(1, $revision_field_data['table']['join']);
-    $this->assertEquals([
-      'entity_test_mulrev_field_data' => [
-        'left_field' => 'revision_id',
-        'field' => 'revision_id',
-        'type' => 'INNER',
-      ],
-    ], $revision_field_data['table']['join']);
-
-    $revision_base_data = $data['entity_test_mulrev_revision'];
-    $this->assertCount(1, $revision_base_data['table']['join']);
-    $this->assertEquals([
-      'entity_test_mulrev_property_revision' => [
-        'left_field' => 'revision_id',
-        'field' => 'revision_id',
-        'type' => 'INNER',
-      ],
-    ], $revision_base_data['table']['join']);
-    $this->assertFalse(isset($data['data_table']));
-  }
-
-  /**
-   * Helper method to mock all store definitions.
-   */
-  protected function setupFieldStorageDefinition() {
-    $id_field_storage_definition = $this->createMock('Drupal\Core\Field\FieldStorageDefinitionInterface');
-    $id_field_storage_definition->expects($this->any())
-      ->method('getSchema')
-      ->willReturn(IntegerItem::schema($id_field_storage_definition));
-    $uuid_field_storage_definition = $this->createMock('Drupal\Core\Field\FieldStorageDefinitionInterface');
-    $uuid_field_storage_definition->expects($this->any())
-      ->method('getSchema')
-      ->willReturn(UuidItem::schema($uuid_field_storage_definition));
-    $type_field_storage_definition = $this->createMock('Drupal\Core\Field\FieldStorageDefinitionInterface');
-    $type_field_storage_definition->expects($this->any())
-      ->method('getSchema')
-      ->willReturn(StringItem::schema($type_field_storage_definition));
-    $langcode_field_storage_definition = $this->createMock('Drupal\Core\Field\FieldStorageDefinitionInterface');
-    $langcode_field_storage_definition->expects($this->any())
-      ->method('getSchema')
-      ->willReturn(LanguageItem::schema($langcode_field_storage_definition));
-    $name_field_storage_definition = $this->createMock('Drupal\Core\Field\FieldStorageDefinitionInterface');
-    $name_field_storage_definition->expects($this->any())
-      ->method('getSchema')
-      ->willReturn(StringItem::schema($name_field_storage_definition));
-    $description_field_storage_definition = $this->createMock('Drupal\Core\Field\FieldStorageDefinitionInterface');
-    $description_field_storage_definition->expects($this->any())
-      ->method('getSchema')
-      ->willReturn(TextLongItem::schema($description_field_storage_definition));
-    $homepage_field_storage_definition = $this->createMock('Drupal\Core\Field\FieldStorageDefinitionInterface');
-    $homepage_field_storage_definition->expects($this->any())
-      ->method('getSchema')
-      ->willReturn(UriItem::schema($homepage_field_storage_definition));
-    $string_field_storage_definition = $this->createMock('Drupal\Core\Field\FieldStorageDefinitionInterface');
-    $string_field_storage_definition->expects($this->any())
-      ->method('getSchema')
-      ->willReturn(StringItem::schema($string_field_storage_definition));
-
-    // Setup the user_id entity reference field.
-    $this->entityTypeManager->expects($this->any())
-      ->method('getDefinition')
-      ->willReturnMap([
-          ['user', TRUE, static::userEntityInfo()],
-        ]
-      );
-    $this->entityTypeManager->expects($this->any())
-      ->method('getDefinition')
-      ->willReturnMap([
-          ['user', TRUE, static::userEntityInfo()],
-        ]
-      );
-    $user_id_field_storage_definition = $this->createMock('Drupal\Core\Field\FieldStorageDefinitionInterface');
-    $user_id_field_storage_definition->expects($this->any())
-      ->method('getSetting')
-      ->with('target_type')
-      ->willReturn('user');
-    $user_id_field_storage_definition->expects($this->any())
-      ->method('getSettings')
-      ->willReturn(['target_type' => 'user']);
-    $user_id_field_storage_definition->expects($this->any())
-      ->method('getSchema')
-      ->willReturn(EntityReferenceItem::schema($user_id_field_storage_definition));
-
-    $revision_id_field_storage_definition = $this->createMock('Drupal\Core\Field\FieldStorageDefinitionInterface');
-    $revision_id_field_storage_definition->expects($this->any())
-      ->method('getSchema')
-      ->willReturn(IntegerItem::schema($revision_id_field_storage_definition));
-
-    $this->entityFieldManager->expects($this->any())
-      ->method('getFieldStorageDefinitions')
-      ->willReturn([
-        'id' => $id_field_storage_definition,
-        'uuid' => $uuid_field_storage_definition,
-        'type' => $type_field_storage_definition,
-        'langcode' => $langcode_field_storage_definition,
-        'name' => $name_field_storage_definition,
-        'description' => $description_field_storage_definition,
-        'homepage' => $homepage_field_storage_definition,
-        'string' => $string_field_storage_definition,
-        'user_id' => $user_id_field_storage_definition,
-        'revision_id' => $revision_id_field_storage_definition,
-      ]);
-  }
-
-  /**
-   * Tests fields on the base table.
-   */
-  public function testBaseTableFields() {
-    $base_field_definitions = $this->setupBaseFields(EntityTest::baseFieldDefinitions($this->baseEntityType));
-    $user_base_field_definitions = [
-      'uid' => BaseFieldDefinition::create('integer')
-        ->setLabel('ID')
-        ->setDescription('The ID of the user entity.')
-        ->setReadOnly(TRUE)
-        ->setSetting('unsigned', TRUE),
-    ];
-    $this->entityFieldManager->expects($this->any())
-      ->method('getBaseFieldDefinitions')
-      ->will($this->returnValueMap([
-        ['user', $user_base_field_definitions],
-        ['entity_test', $base_field_definitions],
-      ]));
-    $this->entityFieldManager->expects($this->any())
-      ->method('getBaseFieldDefinitions')
-      ->will($this->returnValueMap([
-        ['user', $user_base_field_definitions],
-        ['entity_test', $base_field_definitions],
-      ]));
-    // Setup the table mapping.
-    $table_mapping = $this->getMockBuilder(DefaultTableMapping::class)
-      ->disableOriginalConstructor()
-      ->getMock();
-    $table_mapping->expects($this->any())
-      ->method('getTableNames')
-      ->willReturn(['entity_test', 'entity_test__string']);
-    $table_mapping->expects($this->any())
-      ->method('getColumnNames')
-      ->willReturnMap([
-        ['id', ['value' => 'id']],
-        ['uuid', ['value' => 'uuid']],
-        ['type', ['value' => 'type']],
-        ['langcode', ['value' => 'langcode']],
-        ['name', ['value' => 'name']],
-        ['description', ['value' => 'description__value', 'format' => 'description__format']],
-        ['homepage', ['value' => 'homepage']],
-        ['user_id', ['target_id' => 'user_id']],
-        ['string', ['value' => 'string_value']],
-      ]);
-    $table_mapping->expects($this->any())
-      ->method('getFieldNames')
-      ->willReturnMap([
-        ['entity_test', ['id', 'uuid', 'type', 'langcode', 'name', 'description', 'homepage', 'user_id']],
-        ['entity_test__string', ['string']],
-      ]);
-    $table_mapping->expects($this->any())
-      ->method('requiresDedicatedTableStorage')
-      ->willReturnCallback(function (BaseFieldDefinition $base_field) {
-        return $base_field->getName() === 'string';
-      });
-    $table_mapping->expects($this->any())
-      ->method('getDedicatedDataTableName')
-      ->willReturnCallback(function (BaseFieldDefinition $base_field) {
-        if ($base_field->getName() === 'string') {
-          return 'entity_test__string';
-        }
-      });
-
-    $this->entityStorage->expects($this->once())
-      ->method('getTableMapping')
-      ->willReturn($table_mapping);
-
-    $this->setupFieldStorageDefinition();
-
-    $data = $this->viewsData->getViewsData();
-
-    $this->assertNumericField($data['entity_test']['id']);
-    $this->assertField($data['entity_test']['id'], 'id');
-    $this->assertUuidField($data['entity_test']['uuid']);
-    $this->assertField($data['entity_test']['uuid'], 'uuid');
-    $this->assertStringField($data['entity_test']['type']);
-    $this->assertEquals('type', $data['entity_test']['type']['entity field']);
-
-    $this->assertLanguageField($data['entity_test']['langcode']);
-    $this->assertField($data['entity_test']['langcode'], 'langcode');
-    $this->assertEquals('Original language', $data['entity_test']['langcode']['title']);
-
-    $this->assertStringField($data['entity_test']['name']);
-    $this->assertField($data['entity_test']['name'], 'name');
-
-    $this->assertLongTextField($data['entity_test'], 'description');
-    $this->assertField($data['entity_test']['description__value'], 'description');
-    $this->assertField($data['entity_test']['description__format'], 'description');
-
-    $this->assertUriField($data['entity_test']['homepage']);
-    $this->assertField($data['entity_test']['homepage'], 'homepage');
-
-    $this->assertEntityReferenceField($data['entity_test']['user_id']);
-    $this->assertField($data['entity_test']['user_id'], 'user_id');
-
-    $relationship = $data['entity_test']['user_id']['relationship'];
-    $this->assertEquals('users_field_data', $relationship['base']);
-    $this->assertEquals('uid', $relationship['base field']);
-
-    // The string field name should be used as the 'entity field' but the actual
-    // field should reflect what the column mapping is using for multi-value
-    // base fields NOT just the field name. The actual column name returned from
-    // mappings in the test mocks is 'value'.
-    $this->assertStringField($data['entity_test__string']['string_value']);
-    $this->assertField($data['entity_test__string']['string_value'], 'string');
-    $this->assertEquals([
-      'left_field' => 'id',
-      'field' => 'entity_id',
-      'extra' => [[
-          'field' => 'deleted',
-          'value' => 0,
-          'numeric' => TRUE,
-        ],
-      ],
-    ], $data['entity_test__string']['table']['join']['entity_test']);
-  }
-
-  /**
-   * Tests fields on the data table.
-   */
-  public function testDataTableFields() {
-    $entity_type = $this->baseEntityType
-      ->set('data_table', 'entity_test_mul_property_data')
-      ->set('base_table', 'entity_test_mul')
-      ->set('id', 'entity_test_mul')
-      ->setKey('bundle', 'type');
-    $base_field_definitions = $this->setupBaseFields(EntityTestMul::baseFieldDefinitions($this->baseEntityType));
-    $base_field_definitions['type'] = BaseFieldDefinition::create('entity_reference')
-      ->setLabel('entity test type')
-      ->setSetting('target_type', 'entity_test_bundle')
-      ->setTranslatable(TRUE);
-    $base_field_definitions = $this->setupBaseFields($base_field_definitions);
-    $user_base_field_definitions = [
-      'uid' => BaseFieldDefinition::create('integer')
-        ->setLabel('ID')
-        ->setDescription('The ID of the user entity.')
-        ->setReadOnly(TRUE)
-        ->setSetting('unsigned', TRUE),
-    ];
-    $entity_test_type = new ConfigEntityType(['id' => 'entity_test_bundle']);
-
-    $this->entityFieldManager->expects($this->any())
-      ->method('getBaseFieldDefinitions')
-      ->will($this->returnValueMap([
-        ['user', $user_base_field_definitions],
-        ['entity_test_mul', $base_field_definitions],
-      ]));
-    $this->entityFieldManager->expects($this->any())
-      ->method('getBaseFieldDefinitions')
-      ->will($this->returnValueMap([
-        ['user', $user_base_field_definitions],
-        ['entity_test_mul', $base_field_definitions],
-      ]));
-
-    $this->viewsData->setEntityType($entity_type);
-
-    // Setup the table mapping.
-    $table_mapping = $this->getMockBuilder(DefaultTableMapping::class)
-      ->disableOriginalConstructor()
-      ->getMock();
-    $table_mapping->expects($this->any())
-      ->method('getTableNames')
-      ->willReturn(['entity_test_mul', 'entity_test_mul_property_data', 'entity_test_mul__string']);
-    $table_mapping->expects($this->any())
-      ->method('getColumnNames')
-      ->willReturnMap([
-        ['id', ['value' => 'id']],
-        ['uuid', ['value' => 'uuid']],
-        ['type', ['value' => 'type']],
-        ['langcode', ['value' => 'langcode']],
-        ['name', ['value' => 'name']],
-        ['description', ['value' => 'description__value', 'format' => 'description__format']],
-        ['homepage', ['value' => 'homepage']],
-        ['user_id', ['target_id' => 'user_id']],
-        ['string', ['value' => 'string_value']],
-      ]);
-    $table_mapping->expects($this->any())
-      ->method('getFieldNames')
-      ->willReturnMap([
-        ['entity_test_mul', ['uuid']],
-        ['entity_test_mul_property_data', ['id', 'type', 'langcode', 'name', 'description', 'homepage', 'user_id']],
-        ['entity_test_mul__string', ['string']],
-      ]);
-
-    $table_mapping->expects($this->any())
-      ->method('getFieldTableName')
-      ->willReturnCallback(function ($field) {
-        if ($field == 'uuid') {
-          return 'entity_test_mul';
-        }
-        return 'entity_test_mul_property_data';
-      });
-    $table_mapping->expects($this->any())
-      ->method('requiresDedicatedTableStorage')
-      ->willReturnCallback(function (BaseFieldDefinition $base_field) {
-        return $base_field->getName() === 'string';
-      });
-    $table_mapping->expects($this->any())
-      ->method('getDedicatedDataTableName')
-      ->willReturnCallback(function (BaseFieldDefinition $base_field) {
-        if ($base_field->getName() === 'string') {
-          return 'entity_test_mul__string';
-        }
-      });
-
-    $this->entityStorage->expects($this->once())
-      ->method('getTableMapping')
-      ->willReturn($table_mapping);
-
-    $this->setupFieldStorageDefinition();
-
-    $user_entity_type = static::userEntityInfo();
-    $this->entityTypeManager->expects($this->any())
-      ->method('getDefinition')
-      ->will($this->returnValueMap([
-        ['user', TRUE, $user_entity_type],
-        ['entity_test_bundle', TRUE, $entity_test_type],
-      ]));
-
-    $data = $this->viewsData->getViewsData();
-
-    // Check the base fields.
-    $this->assertFalse(isset($data['entity_test_mul']['id']));
-    $this->assertFalse(isset($data['entity_test_mul']['type']));
-    $this->assertUuidField($data['entity_test_mul']['uuid']);
-    $this->assertField($data['entity_test_mul']['uuid'], 'uuid');
-
-    $this->assertFalse(isset($data['entity_test_mul']['type']['relationship']));
-
-    // Also ensure that field_data only fields don't appear on the base table.
-    $this->assertFalse(isset($data['entity_test_mul']['name']));
-    $this->assertFalse(isset($data['entity_test_mul']['description']));
-    $this->assertFalse(isset($data['entity_test_mul']['description__value']));
-    $this->assertFalse(isset($data['entity_test_mul']['description__format']));
-    $this->assertFalse(isset($data['entity_test_mul']['user_id']));
-    $this->assertFalse(isset($data['entity_test_mul']['homepage']));
-
-    // Check the data fields.
-    $this->assertNumericField($data['entity_test_mul_property_data']['id']);
-    $this->assertField($data['entity_test_mul_property_data']['id'], 'id');
-
-    $this->assertBundleField($data['entity_test_mul_property_data']['type']);
-    $this->assertField($data['entity_test_mul_property_data']['type'], 'type');
-
-    $this->assertLanguageField($data['entity_test_mul_property_data']['langcode']);
-    $this->assertField($data['entity_test_mul_property_data']['langcode'], 'langcode');
-    $this->assertEquals('Translation language', $data['entity_test_mul_property_data']['langcode']['title']);
-
-    $this->assertStringField($data['entity_test_mul_property_data']['name']);
-    $this->assertField($data['entity_test_mul_property_data']['name'], 'name');
-
-    $this->assertLongTextField($data['entity_test_mul_property_data'], 'description');
-    $this->assertField($data['entity_test_mul_property_data']['description__value'], 'description');
-    $this->assertField($data['entity_test_mul_property_data']['description__format'], 'description');
-
-    $this->assertUriField($data['entity_test_mul_property_data']['homepage']);
-    $this->assertField($data['entity_test_mul_property_data']['homepage'], 'homepage');
-
-    $this->assertEntityReferenceField($data['entity_test_mul_property_data']['user_id']);
-    $this->assertField($data['entity_test_mul_property_data']['user_id'], 'user_id');
-    $relationship = $data['entity_test_mul_property_data']['user_id']['relationship'];
-    $this->assertEquals('users_field_data', $relationship['base']);
-    $this->assertEquals('uid', $relationship['base field']);
-
-    $this->assertStringField($data['entity_test_mul__string']['string_value']);
-    $this->assertField($data['entity_test_mul__string']['string_value'], 'string');
-    $this->assertEquals([
-      'left_field' => 'id',
-      'field' => 'entity_id',
-      'extra' => [[
-          'field' => 'deleted',
-          'value' => 0,
-          'numeric' => TRUE,
-        ],
-      ],
-    ], $data['entity_test_mul__string']['table']['join']['entity_test_mul']);
-  }
-
-  /**
-   * Tests fields on the revision table.
-   */
-  public function testRevisionTableFields() {
-    $entity_type = $this->baseEntityType
-      ->set('base_table', 'entity_test_mulrev')
-      ->set('revision_table', 'entity_test_mulrev_revision')
-      ->set('data_table', 'entity_test_mulrev_property_data')
-      ->set('revision_data_table', 'entity_test_mulrev_property_revision')
-      ->set('id', 'entity_test_mulrev')
-      ->set('translatable', TRUE);
-    $base_field_definitions = $this->setupBaseFields(EntityTestMulRev::baseFieldDefinitions($this->baseEntityType));
-    $user_base_field_definitions = [
-      'uid' => BaseFieldDefinition::create('integer')
-        ->setLabel('ID')
-        ->setDescription('The ID of the user entity.')
-        ->setReadOnly(TRUE)
-        ->setSetting('unsigned', TRUE),
-    ];
-    $this->entityFieldManager->expects($this->any())
-      ->method('getBaseFieldDefinitions')
-      ->will($this->returnValueMap([
-        ['user', $user_base_field_definitions],
-        ['entity_test_mulrev', $base_field_definitions],
-      ]));
-    $this->entityFieldManager->expects($this->any())
-      ->method('getBaseFieldDefinitions')
-      ->will($this->returnValueMap([
-        ['user', $user_base_field_definitions],
-        ['entity_test_mulrev', $base_field_definitions],
-      ]));
-
-    $this->viewsData->setEntityType($entity_type);
-
-    // Setup the table mapping.
-    $table_mapping = $this->getMockBuilder(DefaultTableMapping::class)
-      ->disableOriginalConstructor()
-      ->getMock();
-    $table_mapping->expects($this->any())
-      ->method('getTableNames')
-      ->willReturn(['entity_test_mulrev', 'entity_test_mulrev_revision', 'entity_test_mulrev_property_data', 'entity_test_mulrev_property_revision', 'entity_test_mulrev__string', 'entity_test_mulrev_revision__string']);
-    $table_mapping->expects($this->any())
-      ->method('getColumnNames')
-      ->willReturnMap([
-        ['id', ['value' => 'id']],
-        ['uuid', ['value' => 'uuid']],
-        ['type', ['value' => 'type']],
-        ['langcode', ['value' => 'langcode']],
-        ['name', ['value' => 'name']],
-        ['description', ['value' => 'description__value', 'format' => 'description__format']],
-        ['homepage', ['value' => 'homepage']],
-        ['user_id', ['target_id' => 'user_id']],
-        ['revision_id', ['value' => 'revision_id']],
-        ['string', ['value' => 'string_value']],
-      ]);
-    $table_mapping->expects($this->any())
-      ->method('getFieldNames')
-      ->willReturnMap([
-        ['entity_test_mulrev', ['id', 'revision_id', 'uuid', 'type']],
-        ['entity_test_mulrev_revision', ['id', 'revision_id', 'langcode']],
-        ['entity_test_mulrev_property_data', ['id', 'revision_id', 'langcode', 'name', 'description', 'homepage', 'user_id']],
-        ['entity_test_mulrev_property_revision', ['id', 'revision_id', 'langcode', 'name', 'description', 'homepage', 'user_id']],
-        ['entity_test_mulrev__string', ['string']],
-        ['entity_test_mulrev_revision__string', ['string']],
-      ]);
-    $table_mapping->expects($this->any())
-      ->method('requiresDedicatedTableStorage')
-      ->willReturnCallback(function (BaseFieldDefinition $base_field) {
-        return $base_field->getName() === 'string';
-      });
-    $table_mapping->expects($this->any())
-      ->method('getDedicatedDataTableName')
-      ->willReturnCallback(function (BaseFieldDefinition $base_field) {
-        if ($base_field->getName() === 'string') {
-          return 'entity_test_mulrev__string';
-        }
-      });
-
-    $table_mapping->expects($this->any())
-      ->method('getDedicatedRevisionTableName')
-      ->willReturnCallback(function (BaseFieldDefinition $base_field) {
-        if ($base_field->getName() === 'string') {
-          return 'entity_test_mulrev_revision__string';
-        }
-      });
-
-    $table_mapping->expects($this->any())
-      ->method('getFieldTableName')
-      ->willReturnCallback(function ($field) {
-        if ($field == 'uuid') {
-          return 'entity_test_mulrev';
-        }
-        return 'entity_test_mulrev_property_data';
-      });
-
-    $this->entityStorage->expects($this->once())
-      ->method('getTableMapping')
-      ->willReturn($table_mapping);
-
-    $this->setupFieldStorageDefinition();
-
-    $data = $this->viewsData->getViewsData();
-
-    // Check the base fields.
-    $this->assertFalse(isset($data['entity_test_mulrev']['id']));
-    $this->assertFalse(isset($data['entity_test_mulrev']['type']));
-    $this->assertFalse(isset($data['entity_test_mulrev']['revision_id']));
-    $this->assertUuidField($data['entity_test_mulrev']['uuid']);
-    $this->assertField($data['entity_test_mulrev']['uuid'], 'uuid');
-
-    // Also ensure that field_data only fields don't appear on the base table.
-    $this->assertFalse(isset($data['entity_test_mulrev']['name']));
-    $this->assertFalse(isset($data['entity_test_mul']['description']));
-    $this->assertFalse(isset($data['entity_test_mul']['description__value']));
-    $this->assertFalse(isset($data['entity_test_mul']['description__format']));
-    $this->assertFalse(isset($data['entity_test_mul']['homepage']));
-    $this->assertFalse(isset($data['entity_test_mulrev']['langcode']));
-    $this->assertFalse(isset($data['entity_test_mulrev']['user_id']));
-
-    // Check the revision fields. The revision ID should only appear in the data
-    // table.
-    $this->assertFalse(isset($data['entity_test_mulrev_revision']['revision_id']));
-
-    // Also ensure that field_data only fields don't appear on the revision table.
-    $this->assertFalse(isset($data['entity_test_mulrev_revision']['id']));
-    $this->assertFalse(isset($data['entity_test_mulrev_revision']['name']));
-    $this->assertFalse(isset($data['entity_test_mulrev_revision']['description']));
-    $this->assertFalse(isset($data['entity_test_mulrev_revision']['description__value']));
-    $this->assertFalse(isset($data['entity_test_mulrev_revision']['description__format']));
-    $this->assertFalse(isset($data['entity_test_mulrev_revision']['homepage']));
-    $this->assertFalse(isset($data['entity_test_mulrev_revision']['user_id']));
-
-    // Check the data fields.
-    $this->assertNumericField($data['entity_test_mulrev_property_data']['id']);
-    $this->assertField($data['entity_test_mulrev_property_data']['id'], 'id');
-    $this->assertNumericField($data['entity_test_mulrev_property_data']['revision_id']);
-    $this->assertField($data['entity_test_mulrev_property_data']['revision_id'], 'revision_id');
-    $this->assertLanguageField($data['entity_test_mulrev_property_data']['langcode']);
-    $this->assertField($data['entity_test_mulrev_property_data']['langcode'], 'langcode');
-    $this->assertStringField($data['entity_test_mulrev_property_data']['name']);
-    $this->assertField($data['entity_test_mulrev_property_data']['name'], 'name');
-
-    $this->assertLongTextField($data['entity_test_mulrev_property_data'], 'description');
-    $this->assertField($data['entity_test_mulrev_property_data']['description__value'], 'description');
-    $this->assertField($data['entity_test_mulrev_property_data']['description__format'], 'description');
-    $this->assertUriField($data['entity_test_mulrev_property_data']['homepage']);
-    $this->assertField($data['entity_test_mulrev_property_data']['homepage'], 'homepage');
-
-    $this->assertEntityReferenceField($data['entity_test_mulrev_property_data']['user_id']);
-    $this->assertField($data['entity_test_mulrev_property_data']['user_id'], 'user_id');
-    $relationship = $data['entity_test_mulrev_property_data']['user_id']['relationship'];
-    $this->assertEquals('users_field_data', $relationship['base']);
-    $this->assertEquals('uid', $relationship['base field']);
-
-    // Check the property data fields.
-    $this->assertNumericField($data['entity_test_mulrev_property_revision']['id']);
-    $this->assertField($data['entity_test_mulrev_property_revision']['id'], 'id');
-
-    $this->assertLanguageField($data['entity_test_mulrev_property_revision']['langcode']);
-    $this->assertField($data['entity_test_mulrev_property_revision']['langcode'], 'langcode');
-    $this->assertEquals('Translation language', $data['entity_test_mulrev_property_revision']['langcode']['title']);
-
-    $this->assertStringField($data['entity_test_mulrev_property_revision']['name']);
-    $this->assertField($data['entity_test_mulrev_property_revision']['name'], 'name');
-
-    $this->assertLongTextField($data['entity_test_mulrev_property_revision'], 'description');
-    $this->assertField($data['entity_test_mulrev_property_revision']['description__value'], 'description');
-    $this->assertField($data['entity_test_mulrev_property_revision']['description__format'], 'description');
-
-    $this->assertUriField($data['entity_test_mulrev_property_revision']['homepage']);
-    $this->assertField($data['entity_test_mulrev_property_revision']['homepage'], 'homepage');
-
-    $this->assertEntityReferenceField($data['entity_test_mulrev_property_revision']['user_id']);
-    $this->assertField($data['entity_test_mulrev_property_revision']['user_id'], 'user_id');
-    $relationship = $data['entity_test_mulrev_property_revision']['user_id']['relationship'];
-    $this->assertEquals('users_field_data', $relationship['base']);
-    $this->assertEquals('uid', $relationship['base field']);
-
-    $this->assertStringField($data['entity_test_mulrev__string']['string_value']);
-    $this->assertField($data['entity_test_mulrev__string']['string_value'], 'string');
-    $this->assertEquals([
-      'left_field' => 'id',
-      'field' => 'entity_id',
-      'extra' => [[
-          'field' => 'deleted',
-          'value' => 0,
-          'numeric' => TRUE,
-        ],
-      ],
-    ], $data['entity_test_mulrev__string']['table']['join']['entity_test_mulrev_property_data']);
-
-    $this->assertStringField($data['entity_test_mulrev_revision__string']['string_value']);
-    $this->assertField($data['entity_test_mulrev_revision__string']['string_value'], 'string');
-    $this->assertEquals([
-      'left_field' => 'revision_id',
-      'field' => 'entity_id',
-      'extra' => [[
-          'field' => 'deleted',
-          'value' => 0,
-          'numeric' => TRUE,
-        ],
-      ],
-    ], $data['entity_test_mulrev_revision__string']['table']['join']['entity_test_mulrev_property_revision']);
-  }
-
-  /**
-   * Tests generic stuff per field.
-   *
-   * @param array $data
-   *   The views data to check.
-   * @param string $field_name
-   *   The entity field name.
-   */
-  protected function assertField($data, $field_name) {
-    $this->assertEquals($field_name, $data['entity field']);
-  }
-
-  /**
-   * Tests add link types.
-   */
-  public function testEntityLinks() {
-    $this->baseEntityType->setLinkTemplate('canonical', '/entity_test/{entity_test}');
-    $this->baseEntityType->setLinkTemplate('edit-form', '/entity_test/{entity_test}/edit');
-    $this->baseEntityType->setLinkTemplate('delete-form', '/entity_test/{entity_test}/delete');
-
-    $data = $this->viewsData->getViewsData();
-    foreach (['entity_test', 'entity_test_revision'] as $table_name) {
-      $this->assertEquals('entity_link', $data[$table_name]['view_entity_test']['field']['id']);
-      $this->assertEquals('entity_link_edit', $data[$table_name]['edit_entity_test']['field']['id']);
-      $this->assertEquals('entity_link_delete', $data[$table_name]['delete_entity_test']['field']['id']);
-    }
-  }
-
-  /**
-   * Tests additional edit links.
-   */
-  public function testEntityLinksJustEditForm() {
-    $this->baseEntityType->setLinkTemplate('edit-form', '/entity_test/{entity_test}/edit');
-
-    $data = $this->viewsData->getViewsData();
-
-    foreach (['entity_test', 'entity_test_revision'] as $table_name) {
-      $this->assertFalse(isset($data[$table_name]['view_entity_test']));
-      $this->assertFalse(isset($data[$table_name]['delete_entity_test']));
-
-      $this->assertEquals('entity_link_edit', $data[$table_name]['edit_entity_test']['field']['id']);
-    }
-  }
-
-  /**
-   * @covers ::getViewsData
-   */
-  public function testGetViewsDataWithoutEntityOperations() {
-    // Make sure there is no list builder. The API does not document is
-    // supports resetting entity handlers, so this might break in the future.
-    $this->baseEntityType->setListBuilderClass(NULL);
-    $data = $this->viewsData->getViewsData();
-    $this->assertArrayNotHasKey('operations', $data[$this->baseEntityType->getBaseTable()]);
-  }
-
-  /**
-   * @covers ::getViewsData
-   */
-  public function testGetViewsDataWithEntityOperations() {
-    $this->baseEntityType->setListBuilderClass('\Drupal\Core\Entity\EntityListBuilder');
-    $data = $this->viewsData->getViewsData();
-
-    $tables = ['entity_test', 'entity_test_revision'];
-    $this->assertSame($tables, array_keys($data));
-    foreach ($tables as $table_name) {
-      $this->assertSame('entity_operations', $data[$table_name]['operations']['field']['id']);
-    }
-  }
-
-  /**
-   * @covers ::getViewsData
-   */
-  public function testGetViewsDataWithNonRevisionableEntityOperations() {
-    $this->baseEntityType->setListBuilderClass('\Drupal\Core\Entity\EntityListBuilder');
-
-    $entity_type_without_revisions = $this->baseEntityType;
-    $views_data = $this->viewsData;
-
-    $entity_type_keys = $entity_type_without_revisions->getKeys();
-    unset($entity_type_keys['revision']);
-
-    $entity_type_without_revisions->set('entity_keys', $entity_type_keys);
-    $views_data->setEntityType($entity_type_without_revisions);
-
-    $data = $views_data->getViewsData();
-
-    $tables = ['entity_test'];
-    $this->assertSame($tables, array_keys($data));
-    foreach ($tables as $table_name) {
-      $this->assertSame('entity_operations', $data[$table_name]['operations']['field']['id']);
-    }
-  }
-
-  /**
-   * Tests views data for a string field.
-   *
-   * @param $data
-   *   The views data to check.
-   */
-  protected function assertStringField($data) {
-    $this->assertEquals('field', $data['field']['id']);
-    $this->assertEquals('string', $data['filter']['id']);
-    $this->assertEquals('string', $data['argument']['id']);
-    $this->assertEquals('standard', $data['sort']['id']);
-  }
-
-  /**
-   * Tests views data for a URI field.
-   *
-   * @param $data
-   *   The views data to check.
-   */
-  protected function assertUriField($data) {
-    $this->assertEquals('field', $data['field']['id']);
-    $this->assertEquals('string', $data['field']['default_formatter']);
-    $this->assertEquals('string', $data['filter']['id']);
-    $this->assertEquals('string', $data['argument']['id']);
-    $this->assertEquals('standard', $data['sort']['id']);
-  }
-
-  /**
-   * Tests views data for a long text field.
-   *
-   * @param $data
-   *   The views data for the table this field is in.
-   * @param $field_name
-   *   The name of the field being checked.
-   */
-  protected function assertLongTextField($data, $field_name) {
-    $value_field = $data[$field_name . '__value'];
-    $this->assertEquals('field', $value_field['field']['id']);
-    $this->assertEquals($field_name . '__format', $value_field['field']['format']);
-    $this->assertEquals('string', $value_field['filter']['id']);
-    $this->assertEquals('string', $value_field['argument']['id']);
-    $this->assertEquals('standard', $value_field['sort']['id']);
-
-    $this->assertStringField($data[$field_name . '__format']);
-  }
-
-  /**
-   * Tests views data for a UUID field.
-   *
-   * @param array $data
-   *   The views data to check.
-   */
-  protected function assertUuidField($data) {
-    // @todo Can we provide additional support for UUIDs in views?
-    $this->assertEquals('field', $data['field']['id']);
-    $this->assertFalse($data['field']['click sortable']);
-    $this->assertEquals('string', $data['filter']['id']);
-    $this->assertEquals('string', $data['argument']['id']);
-    $this->assertEquals('standard', $data['sort']['id']);
-  }
-
-  /**
-   * Tests views data for a numeric field.
-   *
-   * @param array $data
-   *   The views data to check.
-   */
-  protected function assertNumericField($data) {
-    $this->assertEquals('field', $data['field']['id']);
-    $this->assertEquals('numeric', $data['filter']['id']);
-    $this->assertEquals('numeric', $data['argument']['id']);
-    $this->assertEquals('standard', $data['sort']['id']);
-  }
-
-  /**
-   * Tests views data for a language field.
-   *
-   * @param array $data
-   *   The views data to check.
-   */
-  protected function assertLanguageField($data) {
-    $this->assertEquals('field', $data['field']['id']);
-    $this->assertEquals('language', $data['filter']['id']);
-    $this->assertEquals('language', $data['argument']['id']);
-    $this->assertEquals('standard', $data['sort']['id']);
-  }
-
-  /**
-   * Tests views data for an entity reference field.
-   */
-  protected function assertEntityReferenceField($data) {
-    $this->assertEquals('field', $data['field']['id']);
-    $this->assertEquals('numeric', $data['filter']['id']);
-    $this->assertEquals('numeric', $data['argument']['id']);
-    $this->assertEquals('standard', $data['sort']['id']);
-  }
-
-  /**
-   * Tests views data for a bundle field.
-   */
-  protected function assertBundleField($data) {
-    $this->assertEquals('field', $data['field']['id']);
-    $this->assertEquals('bundle', $data['filter']['id']);
-    $this->assertEquals('string', $data['argument']['id']);
-    $this->assertEquals('standard', $data['sort']['id']);
-  }
-
-  /**
-   * Returns entity info for the user entity.
-   *
-   * @return array
-   */
-  protected static function userEntityInfo() {
-    return new ContentEntityType([
-      'id' => 'user',
-      'class' => 'Drupal\user\Entity\User',
-      'label' => 'User',
-      'base_table' => 'users',
-      'data_table' => 'users_field_data',
-      'entity_keys' => [
-        'id' => 'uid',
-        'uuid' => 'uuid',
-      ],
-    ]);
-  }
-
-}
-
-class TestEntityViewsData extends EntityViewsData {
-
-  public function setEntityType(EntityTypeInterface $entity_type) {
-    $this->entityType = $entity_type;
-  }
-
-}
-
-class TestEntityType extends ContentEntityType {
-
-  /**
-   * Sets a specific entity key.
-   *
-   * @param string $key
-   *   The name of the entity key.
-   * @param string $value
-   *   The new value of the key.
-   *
-   * @return $this
-   */
-  public function setKey($key, $value) {
-    $this->entity_keys[$key] = $value;
-    return $this;
-  }
-
-}
diff --git a/web/core/tests/Drupal/KernelTests/Core/Entity/EntityQueryAggregateTest.php b/web/core/tests/Drupal/KernelTests/Core/Entity/EntityQueryAggregateTest.php
index b0cfc72804d42f84cf16adb2a1a24351eee0448c..796f6ca1391e61612cabfb54478ad60edd653f7c 100644
--- a/web/core/tests/Drupal/KernelTests/Core/Entity/EntityQueryAggregateTest.php
+++ b/web/core/tests/Drupal/KernelTests/Core/Entity/EntityQueryAggregateTest.php
@@ -139,7 +139,7 @@ public function testAggregation() {
       // We need to check that a character exists before and after the table,
       // column and alias identifiers. These would be the quote characters
       // specific for each database system.
-      $this->assertMatchesRegularExpression('/' . $aggregation_function . '\(.entity_test.\..id.\) AS .id_' . $aggregation_function . './', (string) $query, 'The argument to the aggregation function should be a quoted field.');
+      $this->assertMatchesRegularExpression('/' . $aggregation_function . '\(.*entity_test.\..id.\).* AS .id_' . $aggregation_function . './', (string) $query, 'The argument to the aggregation function should be a quoted field.');
       $this->assertEquals($expected, $this->queryResult);
     }
 
diff --git a/web/core/tests/Drupal/KernelTests/Core/Render/Element/MachineNameTest.php b/web/core/tests/Drupal/KernelTests/Core/Render/Element/MachineNameTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..74c3e95a963b5b4794f16e6c651050deefa62d28
--- /dev/null
+++ b/web/core/tests/Drupal/KernelTests/Core/Render/Element/MachineNameTest.php
@@ -0,0 +1,74 @@
+<?php
+
+namespace Drupal\KernelTests\Core\Render\Element;
+
+use Drupal\Core\Form\FormInterface;
+use Drupal\Core\Form\FormStateInterface;
+use Drupal\KernelTests\KernelTestBase;
+
+/**
+ * @coversDefaultClass \Drupal\Core\Render\Element\MachineName
+ * @group Render
+ */
+class MachineNameTest extends KernelTestBase implements FormInterface {
+
+  /**
+   * {@inheritdoc}
+   */
+  protected static $modules = ['system'];
+
+  /**
+   * {@inheritdoc}
+   */
+  public function getFormId() {
+    return __CLASS__;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function buildForm(array $form, FormStateInterface $form_state) {
+    $element = [
+      '#id' => 'test',
+      '#type' => 'machine_name',
+      '#machine_name' => [
+        'source' => [
+          'test_source',
+        ],
+      ],
+      '#name' => 'test_machine_name',
+      '#default_value' => NULL,
+    ];
+
+    $complete_form = [
+      'test_machine_name' => $element,
+      'test_source' => [
+        '#type' => 'textfield',
+      ],
+    ];
+    return $complete_form;
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function validateForm(array &$form, FormStateInterface $form_state) {
+  }
+
+  /**
+   * {@inheritdoc}
+   */
+  public function submitForm(array &$form, FormStateInterface $form_state) {
+  }
+
+  /**
+   * Tests the order of the machine name field and the source.
+   */
+  public function testMachineNameOrderException() {
+    $this->expectException(\LogicException::class);
+    $this->expectErrorMessage('The machine name element "test_machine_name" is defined before the source element "test_source", it must be defined after or the source element must specify an id.');
+    $form = \Drupal::formBuilder()->getForm($this);
+    $this->render($form);
+  }
+
+}
diff --git a/web/core/tests/Drupal/KernelTests/Core/Theme/ClaroVerticalTabsTest.php b/web/core/tests/Drupal/KernelTests/Core/Theme/ClaroVerticalTabsTest.php
index 6ae639149565575d0dd20058335568defda97233..c880c1f3770bdb39930c0cfc6908ba9b18308344 100644
--- a/web/core/tests/Drupal/KernelTests/Core/Theme/ClaroVerticalTabsTest.php
+++ b/web/core/tests/Drupal/KernelTests/Core/Theme/ClaroVerticalTabsTest.php
@@ -26,9 +26,9 @@ public function testVerticalTabs() {
     $this->config('system.theme')->set('default', 'claro')->save();
 
     $form = [
-      '#parents' => [],
+      '#parents' => ['parent'],
       '#array_parents' => [],
-      '#tree' => FALSE,
+      '#tree' => TRUE,
     ];
 
     $form['vertical_tabs'] = [
@@ -38,7 +38,7 @@ public function testVerticalTabs() {
     $form['vertical_tabs_details'] = [
       '#type' => 'details',
       '#title' => 'Details',
-      '#group' => 'vertical_tabs',
+      '#group' => 'parent][vertical_tabs',
     ];
 
     // Needs to be rendered after the vertical tabs.
@@ -61,9 +61,9 @@ public function testVerticalTabs() {
     // Assert that the vertical tab details has the appropriate class.
     $this->assertCount(1, $this->cssSelect('.vertical-tabs__items details.vertical-tabs__item'));
     // Assert that there is a details element.
-    $this->assertCount(1, $this->cssSelect('#edit-container-details'));
+    $this->assertCount(1, $this->cssSelect('#edit-parent-container-details'));
     // Assert that details element doesn't have the vertical tab classes.
-    $this->assertCount(0, $this->cssSelect('#edit-container-details.vertical-tabs__item'));
+    $this->assertCount(0, $this->cssSelect('#edit-parent-container-details.vertical-tabs__item'));
   }
 
 }
diff --git a/web/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/Functional/ManageGitIgnoreTest.php b/web/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/Functional/ManageGitIgnoreTest.php
index dee14dfcf4a304cfb5b1747b6f7e55a03dac3367..e2fe7da2229d6588b091faa723a737ec5104aece 100644
--- a/web/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/Functional/ManageGitIgnoreTest.php
+++ b/web/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/Functional/ManageGitIgnoreTest.php
@@ -202,7 +202,14 @@ public function testUnmanagedGitIgnoreWhenGitNotAvailable() {
     // executable script named 'git' that always exits with 127, as if git were
     // not found. Note that we run our tests using process isolation, so we do
     // not need to restore the PATH when we are done.
-    $unavailableGitPath = $this->fixtures->binFixtureDir('disable-git-bin');
+    $unavailableGitPath = $sut . '/bin';
+    mkdir($unavailableGitPath);
+    $bash = <<<SH
+#!/bin/bash
+exit 127
+
+SH;
+    file_put_contents($unavailableGitPath . '/git', $bash);
     chmod($unavailableGitPath . '/git', 0755);
     $oldPath = getenv('PATH');
     putenv('PATH=' . $unavailableGitPath . ':' . getenv('PATH'));
diff --git a/web/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/scripts/disable-git-bin/git b/web/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/scripts/disable-git-bin/git
deleted file mode 100755
index 36fa746230ae1c508ab92b249828c15ba2f23ef9..0000000000000000000000000000000000000000
--- a/web/core/tests/Drupal/Tests/Composer/Plugin/Scaffold/fixtures/scripts/disable-git-bin/git
+++ /dev/null
@@ -1,2 +0,0 @@
-#!/bin/bash
-exit 127
diff --git a/web/core/tests/Drupal/Tests/Core/Asset/CssCollectionOptimizerUnitTest.php b/web/core/tests/Drupal/Tests/Core/Asset/CssCollectionOptimizerUnitTest.php
new file mode 100644
index 0000000000000000000000000000000000000000..a27ce061978067d4a6ceb8541a321f4b7a6dc43a
--- /dev/null
+++ b/web/core/tests/Drupal/Tests/Core/Asset/CssCollectionOptimizerUnitTest.php
@@ -0,0 +1,82 @@
+<?php
+
+namespace Drupal\Tests\Core\Asset;
+
+use Drupal\Core\Asset\AssetCollectionGrouperInterface;
+use Drupal\Core\Asset\AssetDumperInterface;
+use Drupal\Core\Asset\AssetOptimizerInterface;
+use Drupal\Core\Asset\CssCollectionOptimizer;
+use Drupal\Core\File\FileSystemInterface;
+use Drupal\Core\State\StateInterface;
+use Drupal\Tests\UnitTestCase;
+
+/**
+ * Tests the CSS asset optimizer.
+ *
+ * @group Asset
+ */
+class CssCollectionOptimizerUnitTest extends UnitTestCase {
+
+  /**
+   * The data from the dumper.
+   *
+   * @var string
+   */
+  protected $dumperData;
+
+  /**
+   * A CSS Collection optimizer.
+   *
+   * @var \Drupal\Core\Asset\AssetCollectionOptimizerInterface
+   */
+  protected $optimizer;
+
+  protected function setUp(): void {
+    parent::setUp();
+    $mock_grouper = $this->createMock(AssetCollectionGrouperInterface::class);
+    $mock_grouper->method('group')
+      ->willReturnCallback(function ($assets) {
+        return [
+          [
+            'items' => $assets,
+            'type' => 'file',
+            'preprocess' => TRUE,
+          ],
+        ];
+      });
+    $mock_optimizer = $this->createMock(AssetOptimizerInterface::class);
+    $mock_optimizer->method('optimize')
+      ->willReturn(
+        file_get_contents(__DIR__ . '/css_test_files/css_input_with_import.css.optimized.css'),
+        file_get_contents(__DIR__ . '/css_test_files/css_subfolder/css_input_with_import.css.optimized.css')
+      );
+    $mock_dumper = $this->createMock(AssetDumperInterface::class);
+    $mock_dumper->method('dump')
+      ->willReturnCallback(function ($css) {
+        $this->dumperData = $css;
+      });
+    $mock_state = $this->createMock(StateInterface::class);
+    $mock_file_system = $this->createMock(FileSystemInterface::class);
+    $this->optimizer = new CssCollectionOptimizer($mock_grouper, $mock_optimizer, $mock_dumper, $mock_state, $mock_file_system);
+  }
+
+  /**
+   * Test that css imports with strange letters do not destroy the css output.
+   */
+  public function testCssImport() {
+    $this->optimizer->optimize([
+      'core/modules/system/tests/modules/common_test/common_test_css_import.css' => [
+        'type' => 'file',
+        'data' => 'core/modules/system/tests/modules/common_test/common_test_css_import.css',
+        'preprocess' => TRUE,
+      ],
+      'core/modules/system/tests/modules/common_test/common_test_css_import_not_preprocessed.css' => [
+        'type' => 'file',
+        'data' => 'core/modules/system/tests/modules/common_test/common_test_css_import.css',
+        'preprocess' => TRUE,
+      ],
+    ]);
+    self::assertEquals(file_get_contents(__DIR__ . '/css_test_files/css_input_with_import.css.optimized.aggregated.css'), $this->dumperData);
+  }
+
+}
diff --git a/web/core/tests/Drupal/Tests/Core/Asset/CssOptimizerUnitTest.php b/web/core/tests/Drupal/Tests/Core/Asset/CssOptimizerUnitTest.php
index dba9fc2ad2b94615730fe12b5b5c13e08781c6ad..669e2218917b8fc16f0cf6d41f83634d6aacbcda 100644
--- a/web/core/tests/Drupal/Tests/Core/Asset/CssOptimizerUnitTest.php
+++ b/web/core/tests/Drupal/Tests/Core/Asset/CssOptimizerUnitTest.php
@@ -61,6 +61,8 @@ public function providerTestOptimize() {
       //   file_create_url(). (https://www.drupal.org/node/1961340)
       // - Imported files that are external (protocol-relative URL or not)
       //   should not be expanded. (https://www.drupal.org/node/2014851)
+      //   Potential forms of @import might also include media queries.
+      //   (https://developer.mozilla.org/en-US/docs/Web/CSS/@import)
       [
         [
           'group' => -100,
@@ -72,7 +74,7 @@ public function providerTestOptimize() {
           'browsers' => ['IE' => TRUE, '!IE' => TRUE],
           'basename' => 'css_input_with_import.css',
         ],
-        str_replace('url(images/icon.png)', 'url(' . file_url_transform_relative(file_create_url($path . 'images/icon.png')) . ')', file_get_contents($absolute_path . 'css_input_with_import.css.optimized.css')),
+        str_replace("url('import1.css')", 'url(' . file_url_transform_relative(file_create_url($path . 'import1.css')) . ')', str_replace('url(images/icon.png)', 'url(' . file_url_transform_relative(file_create_url($path . 'images/icon.png')) . ')', file_get_contents($absolute_path . 'css_input_with_import.css.optimized.css'))),
       ],
       // File. Tests:
       // - Retain comment hacks.
diff --git a/web/core/tests/Drupal/Tests/Core/Asset/css_test_files/css_input_with_import.css b/web/core/tests/Drupal/Tests/Core/Asset/css_test_files/css_input_with_import.css
index 34270ee6966444719889593624a4176fea37f762..d71b91b82007e00b466bb83fc42a18ff4b29ca03 100644
--- a/web/core/tests/Drupal/Tests/Core/Asset/css_test_files/css_input_with_import.css
+++ b/web/core/tests/Drupal/Tests/Core/Asset/css_test_files/css_input_with_import.css
@@ -1,9 +1,25 @@
 
 
-@import "import1.css";
+@import 'import1.css';
 @import "import2.css";
+@import url('import1.css');
+@import url("https://fonts.fontprovider.com/css2?family=Roboto+Mono:wght@300;400&family=Roboto:ital,wght@0,300;0,400;1,300;1,400&display=swap") print;
+@import url(import1.css);
+@import url('import1.css') screen;
 @import url("http://example.com/style.css");
 @import url("//example.com/style.css");
+@import url("https://fonts.fontprovider.com/css2?family=Roboto+Mono:wght@300;400&family=Roboto:ital,wght@0,300;0,400;1,300;1,400&display=swap");
+@import url("http://example.com/style.css") screen and (orientation:landscape);
+@import "http://example.com/style.css" screen;
+@import "http://example.com/style.css" supports(display: table-cell);
+@import "http://example.com/style.css" supports(display: table-cell) screen;
+@import url("http://example.com/style.css") screen and (orientation:landscape);
+@import url("http://example.com/style.css") screen;
+@import url("http://user:pass@example.com/style.css") screen and (orientation:landscape);
+@import url(http://example.com/cus\(t;om.css);
+@import url('http://example.com/cu(st;o)m.css');
+@import url("http://user:pass@example.com/cu(s)t;om.css");
+@import url(http://user:pass@example.com/cu\(s\)t;om.css);
 
 body {
   margin: 0;
@@ -29,4 +45,3 @@ textarea, select {
   font: 1em/160% Verdana, sans-serif;
   color: #494949;
 }
-
diff --git a/web/core/tests/Drupal/Tests/Core/Asset/css_test_files/css_input_with_import.css.optimized.aggregated.css b/web/core/tests/Drupal/Tests/Core/Asset/css_test_files/css_input_with_import.css.optimized.aggregated.css
new file mode 100644
index 0000000000000000000000000000000000000000..5ca58da466734c3048ec3ed45e07637c60ebe275
--- /dev/null
+++ b/web/core/tests/Drupal/Tests/Core/Asset/css_test_files/css_input_with_import.css.optimized.aggregated.css
@@ -0,0 +1,14 @@
+@import url("https://fonts.fontprovider.com/css2?family=Roboto+Mono:wght@300;400&family=Roboto:ital,wght@0,300;0,400;1,300;1,400&display=swap") print;@import url('import1.css') screen;@import url("http://example.com/style.css");@import url("//example.com/style.css");@import url("https://fonts.fontprovider.com/css2?family=Roboto+Mono:wght@300;400&family=Roboto:ital,wght@0,300;0,400;1,300;1,400&display=swap");@import url("http://example.com/style.css") screen and (orientation:landscape);@import "http://example.com/style.css" screen;@import "http://example.com/style.css" supports(display:table-cell);@import "http://example.com/style.css" supports(display:table-cell) screen;@import url("http://example.com/style.css") screen and (orientation:landscape);@import url("http://example.com/style.css") screen;@import url("http://user:pass@example.com/style.css") screen and (orientation:landscape);@import url(http://example.com/cus\(t;om.css);@import url('http://example.com/cu(st;o)m.css');@import url("http://user:pass@example.com/cu(s)t;om.css");@import url(http://user:pass@example.com/cu\(s\)t;om.css);ul,select{font:1em/160% Verdana,sans-serif;color:#494949;}.ui-icon{background-image:url(images/icon.png);}.data .double-quote{background-image:url("");}.data .single-quote{background-image:url('');}.data .no-quote{background-image:url();}
+p,select{font:1em/160% Verdana,sans-serif;color:#494949;}
+ul,select{font:1em/160% Verdana,sans-serif;color:#494949;}.ui-icon{background-image:url(images/icon.png);}.data .double-quote{background-image:url("");}.data .single-quote{background-image:url('');}.data .no-quote{background-image:url();}
+ul,select{font:1em/160% Verdana,sans-serif;color:#494949;}.ui-icon{background-image:url(images/icon.png);}.data .double-quote{background-image:url("");}.data .single-quote{background-image:url('');}.data .no-quote{background-image:url();}
+body{margin:0;padding:0;background:#edf5fa;font:76%/170% Verdana,sans-serif;color:#494949;}.this .is .a .test{font:1em/100% Verdana,sans-serif;color:#494949;}.this
+.is
+.a
+.test{font:1em/100% Verdana,sans-serif;color:#494949;}textarea,select{font:1em/160% Verdana,sans-serif;color:#494949;}
+ul,select{font:1em/160% Verdana,sans-serif;color:#494949;}.ui-icon{background-image:url(../images/icon.png);}.data .double-quote{background-image:url("");}.data .single-quote{background-image:url('');}.data .no-quote{background-image:url();}
+p,select{font:1em/160% Verdana,sans-serif;color:#494949;}
+body{margin:0;padding:0;background:#edf5fa;font:76%/170% Verdana,sans-serif;color:#494949;}.this .is .a .test{font:1em/100% Verdana,sans-serif;color:#494949;}.this
+.is
+.a
+.test{font:1em/100% Verdana,sans-serif;color:#494949;}textarea,select{font:1em/160% Verdana,sans-serif;color:#494949;}
diff --git a/web/core/tests/Drupal/Tests/Core/Asset/css_test_files/css_input_with_import.css.optimized.css b/web/core/tests/Drupal/Tests/Core/Asset/css_test_files/css_input_with_import.css.optimized.css
index 16bd93be7ede890e186df3573dc878a81f12f10f..a35ef29adb1dab70c32bbd9533b551a1e272fef1 100644
--- a/web/core/tests/Drupal/Tests/Core/Asset/css_test_files/css_input_with_import.css.optimized.css
+++ b/web/core/tests/Drupal/Tests/Core/Asset/css_test_files/css_input_with_import.css.optimized.css
@@ -1,6 +1,8 @@
 ul,select{font:1em/160% Verdana,sans-serif;color:#494949;}.ui-icon{background-image:url(images/icon.png);}.data .double-quote{background-image:url("");}.data .single-quote{background-image:url('');}.data .no-quote{background-image:url();}
 p,select{font:1em/160% Verdana,sans-serif;color:#494949;}
-@import url("http://example.com/style.css");@import url("//example.com/style.css");body{margin:0;padding:0;background:#edf5fa;font:76%/170% Verdana,sans-serif;color:#494949;}.this .is .a .test{font:1em/100% Verdana,sans-serif;color:#494949;}.this
+ul,select{font:1em/160% Verdana,sans-serif;color:#494949;}.ui-icon{background-image:url(images/icon.png);}.data .double-quote{background-image:url("");}.data .single-quote{background-image:url('');}.data .no-quote{background-image:url();}
+@import url("https://fonts.fontprovider.com/css2?family=Roboto+Mono:wght@300;400&family=Roboto:ital,wght@0,300;0,400;1,300;1,400&display=swap") print;ul,select{font:1em/160% Verdana,sans-serif;color:#494949;}.ui-icon{background-image:url(images/icon.png);}.data .double-quote{background-image:url("");}.data .single-quote{background-image:url('');}.data .no-quote{background-image:url();}
+@import url('import1.css') screen;@import url("http://example.com/style.css");@import url("//example.com/style.css");@import url("https://fonts.fontprovider.com/css2?family=Roboto+Mono:wght@300;400&family=Roboto:ital,wght@0,300;0,400;1,300;1,400&display=swap");@import url("http://example.com/style.css") screen and (orientation:landscape);@import "http://example.com/style.css" screen;@import "http://example.com/style.css" supports(display:table-cell);@import "http://example.com/style.css" supports(display:table-cell) screen;@import url("http://example.com/style.css") screen and (orientation:landscape);@import url("http://example.com/style.css") screen;@import url("http://user:pass@example.com/style.css") screen and (orientation:landscape);@import url(http://example.com/cus\(t;om.css);@import url('http://example.com/cu(st;o)m.css');@import url("http://user:pass@example.com/cu(s)t;om.css");@import url(http://user:pass@example.com/cu\(s\)t;om.css);body{margin:0;padding:0;background:#edf5fa;font:76%/170% Verdana,sans-serif;color:#494949;}.this .is .a .test{font:1em/100% Verdana,sans-serif;color:#494949;}.this
 .is
 .a
 .test{font:1em/100% Verdana,sans-serif;color:#494949;}textarea,select{font:1em/160% Verdana,sans-serif;color:#494949;}
diff --git a/web/core/tests/Drupal/Tests/Core/Logger/LogMessageParserTest.php b/web/core/tests/Drupal/Tests/Core/Logger/LogMessageParserTest.php
index 996ea54c2fb71f43281cf8231340aa669676dcc2..6e45055b512bd4aeaa74adaf64336b29c2df911a 100644
--- a/web/core/tests/Drupal/Tests/Core/Logger/LogMessageParserTest.php
+++ b/web/core/tests/Drupal/Tests/Core/Logger/LogMessageParserTest.php
@@ -60,8 +60,8 @@ public function providerTestParseMessagePlaceholders() {
       ],
       // Message with double PSR3 style messages.
       [
-        ['message' => 'Test {with} two {encapsuled} strings', 'context' => ['with' => 'together', 'encapsuled' => 'awesome']],
-        ['message' => 'Test @with two @encapsuled strings', 'context' => ['@with' => 'together', '@encapsuled' => 'awesome']],
+        ['message' => 'Test {with} two {{encapsuled}} strings', 'context' => ['with' => 'together', 'encapsuled' => 'awesome']],
+        ['message' => 'Test @with two {@encapsuled} strings', 'context' => ['@with' => 'together', '@encapsuled' => 'awesome']],
       ],
     ];
   }
diff --git a/web/core/themes/claro/claro.theme b/web/core/themes/claro/claro.theme
index 5916898c02c8bd99e8308fc6c3e4534625db3633..ba41e6d7c86c95354a5e8442404c7580f2a53074 100644
--- a/web/core/themes/claro/claro.theme
+++ b/web/core/themes/claro/claro.theme
@@ -747,10 +747,12 @@ function claro_form_system_modules_alter(&$form, FormStateInterface $form_state)
       foreach (Element::children($form['modules'][$key]) as $module_key) {
         if (isset($form['modules'][$key][$module_key]['links'])) {
           foreach ($form['modules'][$key][$module_key]['links'] as $link_key => &$link) {
-            $action_link_type = $link_key_to_action_link_type[$link_key];
-            $link['#options']['attributes']['class'][] = 'action-link';
-            $link['#options']['attributes']['class'][] = 'action-link--small';
-            $link['#options']['attributes']['class'][] = "action-link--icon-$action_link_type";
+            if (array_key_exists($link_key, $link_key_to_action_link_type)) {
+              $action_link_type = $link_key_to_action_link_type[$link_key];
+              $link['#options']['attributes']['class'][] = 'action-link';
+              $link['#options']['attributes']['class'][] = 'action-link--small';
+              $link['#options']['attributes']['class'][] = "action-link--icon-$action_link_type";
+            }
           }
         }
       }
diff --git a/web/core/themes/claro/src/ClaroPreRender.php b/web/core/themes/claro/src/ClaroPreRender.php
index c31505dc2d9c75251a7f37665f1875ca2d377264..40885064e95b2fb2b61276a6b9b442b5715bce4b 100644
--- a/web/core/themes/claro/src/ClaroPreRender.php
+++ b/web/core/themes/claro/src/ClaroPreRender.php
@@ -77,13 +77,10 @@ public static function verticalTabs($element) {
       $last_group_with_child_key = NULL;
       $last_group_with_child_key_last_child_key = NULL;
 
-      // Only iterate through the parents instead of all the group keys.
-      foreach ($element['#parents'] as $group_key) {
-        // Check parents against groups because we are only looking for group
-        // elements.
-        if (!in_array($group_key, $group_keys)) {
-          continue;
-        }
+      $group_key = implode('][', $element['#parents']);
+      // Only check siblings against groups because we are only looking for
+      // group elements.
+      if (in_array($group_key, $group_keys)) {
         $children_keys = Element::children($element['group']['#groups'][$group_key], TRUE);
 
         foreach ($children_keys as $child_key) {
diff --git a/web/core/themes/olivero/css/components/fieldset.css b/web/core/themes/olivero/css/components/fieldset.css
index ce2883e86366368950429a44ccbc3bc65f464007..679eb821d574868ba6914191e6b7f51793aec1a0 100644
--- a/web/core/themes/olivero/css/components/fieldset.css
+++ b/web/core/themes/olivero/css/components/fieldset.css
@@ -49,7 +49,7 @@
   padding-top: 0;
   padding-bottom: 0;
   color: inherit;
-  border: solid 2px #7e96a7;
+  border: solid 2px #5d7585;
   border-radius: 0.1875rem;
   background-color: #fff;
 }
@@ -78,7 +78,7 @@ _:-ms-fullscreen,
   color: inherit;
   border-top-left-radius: 3px;
   border-top-right-radius: 3px;
-  background-color: #7e96a7;
+  background-color: #5d7585;
   font-size: 1.125rem;
   font-weight: 700;
   line-height: 1.6875rem
@@ -130,7 +130,7 @@ _:-ms-fullscreen,
 }
 
 .fieldset__label.is-disabled {
-  color: #7e96a7;
+  color: #5d7585;
 }
 
 .fieldset__description {
@@ -185,7 +185,7 @@ _:-ms-fullscreen,
 }
 
 .fieldset--group .fieldset__legend--visible ~ .fieldset__wrapper {
-  border: solid 2px #7e96a7;
+  border: solid 2px #5d7585;
   border-bottom-right-radius: 3px;
   border-bottom-left-radius: 3px;
 }
diff --git a/web/core/themes/olivero/css/components/fieldset.pcss.css b/web/core/themes/olivero/css/components/fieldset.pcss.css
index be0bcb62bc17cc13d73903db6913956716249c14..35a70b2738f3343c4a6e9ffd03a2e2828df84e8f 100644
--- a/web/core/themes/olivero/css/components/fieldset.pcss.css
+++ b/web/core/themes/olivero/css/components/fieldset.pcss.css
@@ -14,7 +14,7 @@
   padding-inline-start: 0;
   padding-inline-end: 0;
   color: inherit;
-  border: solid 2px var(--color--gray-30);
+  border: solid 2px var(--color--gray-25);
   border-radius: var(--border-radius);
   background-color: var(--color--white);
 }
@@ -42,7 +42,7 @@ _:-ms-fullscreen,
   color: inherit;
   border-top-left-radius: var(--border-radius);
   border-top-right-radius: var(--border-radius);
-  background-color: var(--color--gray-30);
+  background-color: var(--color--gray-25);
   font-size: var(--font-size-l);
   font-weight: 700;
   line-height: var(--line-height-base);
@@ -83,7 +83,7 @@ _:-ms-fullscreen,
 }
 
 .fieldset__label.is-disabled {
-  color: var(--color--gray-30);
+  color: var(--color--gray-25);
 }
 
 .fieldset__description {
@@ -126,7 +126,7 @@ _:-ms-fullscreen,
 }
 
 .fieldset--group .fieldset__legend--visible ~ .fieldset__wrapper {
-  border: solid 2px var(--color--gray-30);
+  border: solid 2px var(--color--gray-25);
   border-bottom-right-radius: var(--border-radius);
   border-bottom-left-radius: var(--border-radius);
 }
diff --git a/web/core/themes/olivero/css/components/form-select.css b/web/core/themes/olivero/css/components/form-select.css
index 0fa278ff3a6c9557fe17d76b064c62d87020c005..e51b026e5d7e2b3b2e60bedce250d6a6eb6e8912 100644
--- a/web/core/themes/olivero/css/components/form-select.css
+++ b/web/core/themes/olivero/css/components/form-select.css
@@ -62,7 +62,7 @@ select:focus {
 
 select {
 
-  /* Disables border of select specifically for ms browser */
+  /* Hides default chevron within Internet Explorer. */
 }
 
 select::-ms-expand {
@@ -99,17 +99,58 @@ select.form-element--small {
 
 select {
 
-  /* Necessary for IE11 to show chevron. */
+  /* Necessary to show chevron in forced colors mode in modern browsers. */
 }
 
-@media screen and (-ms-high-contrast: active) {
+@media (forced-colors: active) {
+
+[dir="ltr"] select {
+    padding-right: 1.125rem
+  }
+
+[dir="rtl"] select {
+    padding-left: 1.125rem
+  }
 
 select {
-    background-image: url("data:image/svg+xml,%3csvg width='18' height='11' viewBox='0 0 18 11' fill='none' xmlns='http://www.w3.org/2000/svg'%3e%3cpath d='M18 1.49699C18 1.35271 17.9279 1.19038 17.8196 1.08216L16.9178 0.18036C16.8096 0.0721439 16.6473 0 16.503 0C16.3587 0 16.1964 0.0721439 16.0882 0.18036L9 7.26854L1.91182 0.18036C1.80361 0.0721439 1.64128 0 1.49699 0C1.33467 0 1.19038 0.0721439 1.08216 0.18036L0.180361 1.08216C0.0721442 1.19038 0 1.35271 0 1.49699C0 1.64128 0.0721442 1.80361 0.180361 1.91182L8.58517 10.3166C8.69339 10.4248 8.85571 10.497 9 10.497C9.14429 10.497 9.30661 10.4248 9.41483 10.3166L17.8196 1.91182C17.9279 1.80361 18 1.64128 18 1.49699Z' fill='%235D7585'/%3e%3c/svg%3e")
+    background-image: none;
+    -webkit-appearance: listbox;
+    -moz-appearance: listbox;
+    appearance: listbox /* Default <select> appearance value for modern browsers. */
+
+    /* Lets browser set <select> appearance to whatever the browser's default is. */
 }
+    @supports ((-webkit-appearance: revert) or (-moz-appearance: revert) or (appearance: revert)) {
 
-    select[multiple] {
-      background-image: none;
+select {
+      -webkit-appearance: revert;
+      -moz-appearance: revert;
+      appearance: revert
+}
+    }
+  }
+
+select {
+
+  /* Necessary for Internet Explorer to show chevron. */
+}
+
+@media screen and (-ms-high-contrast: active) {
+
+[dir="ltr"] select {
+    padding-right: 0
+  }
+
+[dir="rtl"] select {
+    padding-left: 0
+  }
+
+select {
+
+    /* Re-enable default chevron for Internet Explorer. */
+}
+    select::-ms-expand {
+      display: block;
     }
   }
 
diff --git a/web/core/themes/olivero/css/components/form-select.pcss.css b/web/core/themes/olivero/css/components/form-select.pcss.css
index 00a8c2833cfa26ad794b04192801e173a1fcdc83..87eba9ab62790893fb012d3c8367070b63b10e30 100644
--- a/web/core/themes/olivero/css/components/form-select.pcss.css
+++ b/web/core/themes/olivero/css/components/form-select.pcss.css
@@ -38,7 +38,7 @@ select {
     }
   }
 
-  /* Disables border of select specifically for ms browser */
+  /* Hides default chevron within Internet Explorer. */
   &::-ms-expand {
     display: none;
   }
@@ -71,12 +71,25 @@ select {
     height: var(--sp2-5);
   }
 
-  /* Necessary for IE11 to show chevron. */
+  /* Necessary to show chevron in forced colors mode in modern browsers. */
+  @media (forced-colors: active) {
+    padding-inline-end: var(--sp);
+    background-image: none;
+    appearance: listbox; /* Default <select> appearance value for modern browsers. */
+
+    /* Lets browser set <select> appearance to whatever the browser's default is. */
+    @supports (appearance: revert) {
+      appearance: revert;
+    }
+  }
+
+  /* Necessary for Internet Explorer to show chevron. */
   @media screen and (-ms-high-contrast: active) {
-    background-image: var(--form-element-select-icon);
+    padding-inline-end: 0;
 
-    &[multiple] {
-      background-image: none;
+    /* Re-enable default chevron for Internet Explorer. */
+    &::-ms-expand {
+      display: block;
     }
   }
 }
diff --git a/web/core/themes/olivero/css/components/header-search-wide.css b/web/core/themes/olivero/css/components/header-search-wide.css
index 6d86cf313d6530627063ba85f1c449654d5b75c9..53c9ac63524aa155ec9473ab68cd31e13c71e84b 100644
--- a/web/core/themes/olivero/css/components/header-search-wide.css
+++ b/web/core/themes/olivero/css/components/header-search-wide.css
@@ -402,6 +402,17 @@
     margin-left: auto;
 }
 
+@media (forced-colors: active) {
+
+.block-search-wide__button {
+    background: ButtonFace
+}
+
+    .block-search-wide__button path {
+      fill: ButtonText;
+    }
+  }
+
 /* Provide rudimentary access to site search if JS is disabled. */
 
 html:not(.js) .search-block-form:focus-within .block-search-wide__wrapper {
diff --git a/web/core/themes/olivero/css/components/header-search-wide.pcss.css b/web/core/themes/olivero/css/components/header-search-wide.pcss.css
index ab5adf95ba90da547c0d0140359e804322e50416..bac07058c78a9e4aa445b047048aa261661de37b 100644
--- a/web/core/themes/olivero/css/components/header-search-wide.pcss.css
+++ b/web/core/themes/olivero/css/components/header-search-wide.pcss.css
@@ -256,6 +256,14 @@
     margin-inline-start: auto;
     margin-inline-end: auto;
   }
+
+  @media (forced-colors: active) {
+    background: ButtonFace;
+
+    & path {
+      fill: ButtonText;
+    }
+  }
 }
 
 /* Provide rudimentary access to site search if JS is disabled. */
diff --git a/web/core/themes/olivero/css/components/site-header.css b/web/core/themes/olivero/css/components/site-header.css
index 879cdd564dfa05a95b62589aa340674ab3c0d637..bc9a9fdd1e6feecc2af42a0769fef8c7923e3ac4 100644
--- a/web/core/themes/olivero/css/components/site-header.css
+++ b/web/core/themes/olivero/css/components/site-header.css
@@ -68,6 +68,7 @@
 }
 
 .site-header__inner {
+  z-index: 1; /* Appear in front of Drupal's tabs. */
   flex-grow: 1;
   width: calc(100vw - 5.625rem);
   background: #fff;
diff --git a/web/core/themes/olivero/css/components/site-header.pcss.css b/web/core/themes/olivero/css/components/site-header.pcss.css
index 0a4d69cd2f484888400a84cd49be2226c83a1dc2..7bc651315c4a18ad8822d1845b35e07c29878939 100644
--- a/web/core/themes/olivero/css/components/site-header.pcss.css
+++ b/web/core/themes/olivero/css/components/site-header.pcss.css
@@ -62,6 +62,7 @@
 }
 
 .site-header__inner {
+  z-index: 1; /* Appear in front of Drupal's tabs. */
   flex-grow: 1;
   width: calc(100vw - var(--content-left));
   background: var(--color--white);
diff --git a/web/core/themes/olivero/css/components/vertical-tabs.css b/web/core/themes/olivero/css/components/vertical-tabs.css
index 7793996c04645ad81bd70d339872da0aac50d6c3..588be625dab2166ed7cf9b3037a01c367eccd16c 100644
--- a/web/core/themes/olivero/css/components/vertical-tabs.css
+++ b/web/core/themes/olivero/css/components/vertical-tabs.css
@@ -105,6 +105,7 @@
   padding-top: 0.5625rem;
   padding-bottom: 0.5625rem;
   text-decoration: none;
+  color: #0f6292;
   background-color: #e7edf1
 }
 
diff --git a/web/core/themes/olivero/css/components/vertical-tabs.pcss.css b/web/core/themes/olivero/css/components/vertical-tabs.pcss.css
index 52e0b73fa45a5ccbc2f0e8a4f722af3a1115607b..be8f2d9d5b1551dd936f5cff101e0b5fa7185043 100644
--- a/web/core/themes/olivero/css/components/vertical-tabs.pcss.css
+++ b/web/core/themes/olivero/css/components/vertical-tabs.pcss.css
@@ -68,6 +68,7 @@
   padding-inline-start: var(--sp0-75);
   padding-inline-end: var(--sp0-75);
   text-decoration: none;
+  color: var(--color--blue-10);
   background-color: var(--color--gray-80);
 
   &:focus,