From 1b82190349b21d8e987ba65b5f78a94aefc2a722 Mon Sep 17 00:00:00 2001
From: xjm <xjm@65776.no-reply.drupal.org>
Date: Tue, 10 Mar 2020 16:32:33 -0500
Subject: [PATCH] Issue #2917600 by tedbow, alexpott, catch, anthonyf, xjm,
 andypost, Alan D., moshe weitzman, Berdir: update_fix_compatibility() puts
 sites into unrecoverable state

---
 core/includes/install.inc                     |   8 +-
 core/includes/update.inc                      |   9 +-
 .../Drupal/Core/Extension/ExtensionList.php   |  17 +
 .../src/Controller/DbUpdateController.php     |   1 -
 core/modules/system/system.install            | 173 +++++++++-
 .../UpdateSystem/UpdateScriptTest.php         | 297 ++++++++++++++++++
 .../Core/Update/CompatibilityFixTest.php      |   4 +
 .../Core/Extension/ExtensionListTest.php      |  72 ++++-
 8 files changed, 566 insertions(+), 15 deletions(-)

diff --git a/core/includes/install.inc b/core/includes/install.inc
index 457dc50bafba..544101ab99d7 100644
--- a/core/includes/install.inc
+++ b/core/includes/install.inc
@@ -78,9 +78,13 @@
  * Loads .install files for installed modules to initialize the update system.
  */
 function drupal_load_updates() {
+  /** @var \Drupal\Core\Extension\ModuleExtensionList $extension_list_module */
+  $extension_list_module = \Drupal::service('extension.list.module');
   foreach (drupal_get_installed_schema_version(NULL, FALSE, TRUE) as $module => $schema_version) {
-    if ($schema_version > -1) {
-      module_load_install($module);
+    if ($extension_list_module->exists($module) && !$extension_list_module->checkIncompatibility($module)) {
+      if ($schema_version > -1) {
+        module_load_install($module);
+      }
     }
   }
 }
diff --git a/core/includes/update.inc b/core/includes/update.inc
index 5254743c4fa1..7cacba384d3a 100644
--- a/core/includes/update.inc
+++ b/core/includes/update.inc
@@ -14,8 +14,13 @@
 
 /**
  * Disables any extensions that are incompatible with the current core version.
+ *
+ * @deprecated in Drupal 8.8.4 and is removed from Drupal 9.0.0.
+ *
+ * @see https://www.drupal.org/node/3026100
  */
 function update_fix_compatibility() {
+  @trigger_error(__FUNCTION__ . '() is deprecated in Drupal 8.8.4 and will be removed before Drupal 9.0.0. There is no replacement. See https://www.drupal.org/node/3026100', E_USER_DEPRECATED);
   // Fix extension objects if the update is being done via Drush 8. In non-Drush
   // environments this will already be fixed by the UpdateKernel this point.
   UpdateKernel::fixSerializedExtensionObjects(\Drupal::getContainer());
@@ -306,9 +311,11 @@ function update_get_update_list() {
   $ret = ['system' => []];
 
   $modules = drupal_get_installed_schema_version(NULL, FALSE, TRUE);
+  /** @var \Drupal\Core\Extension\ExtensionList $extension_list */
+  $extension_list = \Drupal::service('extension.list.module');
   foreach ($modules as $module => $schema_version) {
     // Skip uninstalled and incompatible modules.
-    if ($schema_version == SCHEMA_UNINSTALLED || update_check_incompatibility($module)) {
+    if ($schema_version == SCHEMA_UNINSTALLED || $extension_list->checkIncompatibility($module)) {
       continue;
     }
     // Display a requirements error if the user somehow has a schema version
diff --git a/core/lib/Drupal/Core/Extension/ExtensionList.php b/core/lib/Drupal/Core/Extension/ExtensionList.php
index 8fc8da0ca045..a6c28a49a06e 100644
--- a/core/lib/Drupal/Core/Extension/ExtensionList.php
+++ b/core/lib/Drupal/Core/Extension/ExtensionList.php
@@ -563,4 +563,21 @@ protected function createExtensionInfo(Extension $extension) {
     return $info;
   }
 
+  /**
+   * Tests the compatibility of an extension.
+   *
+   * @param string $name
+   *   The extension name to check.
+   *
+   * @return bool
+   *   TRUE if the extension is incompatible and FALSE if not.
+   *
+   * @throws \Drupal\Core\Extension\Exception\UnknownExtensionException
+   *   If there is no extension with the supplied name.
+   */
+  public function checkIncompatibility($name) {
+    $extension = $this->get($name);
+    return $extension->info['core_incompatible'] || (isset($extension->info['php']) && version_compare(phpversion(), $extension->info['php']) < 0);
+  }
+
 }
diff --git a/core/modules/system/src/Controller/DbUpdateController.php b/core/modules/system/src/Controller/DbUpdateController.php
index 8f29f798261d..2d1e88605fa5 100644
--- a/core/modules/system/src/Controller/DbUpdateController.php
+++ b/core/modules/system/src/Controller/DbUpdateController.php
@@ -144,7 +144,6 @@ public function handle($op, Request $request) {
     require_once $this->root . '/core/includes/update.inc';
 
     drupal_load_updates();
-    update_fix_compatibility();
 
     if ($request->query->get('continue')) {
       $_SESSION['update_ignore_warnings'] = TRUE;
diff --git a/core/modules/system/system.install b/core/modules/system/system.install
index 8db3d323679e..0ea442bf6139 100644
--- a/core/modules/system/system.install
+++ b/core/modules/system/system.install
@@ -29,6 +29,9 @@
  */
 function system_requirements($phase) {
   global $install_state;
+  // Reset the extension lists.
+  \Drupal::service('extension.list.module')->reset();
+  \Drupal::service('extension.list.theme')->reset();
   $requirements = [];
 
   // Report Drupal version
@@ -834,28 +837,71 @@ function system_requirements($phase) {
   }
 
   // Display an error if a newly introduced dependency in a module is not resolved.
-  if ($phase == 'update') {
+  if ($phase === 'update' || $phase === 'runtime') {
+    $create_extension_incompatibility_list = function ($extension_names, $description, $title) {
+      // Use an inline twig template to:
+      // - Concatenate two MarkupInterface objects and preserve safeness.
+      // - Use the item_list theme for the extension list.
+      $template = [
+        '#type' => 'inline_template',
+        '#template' => '{{ description }}{{ extensions }}',
+        '#context' => [
+          'extensions' => [
+            '#theme' => 'item_list',
+          ],
+        ],
+      ];
+      $template['#context']['extensions']['#items'] = $extension_names;
+      $template['#context']['description'] = $description;
+      return [
+        'title' => $title,
+        'value' => [
+          'list' => $template,
+          'handbook_link' => [
+            '#markup' => t(
+              'Review the <a href=":url"> suggestions for resolving this incompatibility</a> to repair your installation, and then re-run update.php.',
+              [':url' => 'https://www.drupal.org/docs/8/update/troubleshooting-database-updates']
+            ),
+          ],
+        ],
+        'severity' => REQUIREMENT_ERROR,
+      ];
+    };
     $profile = \Drupal::installProfile();
     $files = \Drupal::service('extension.list.module')->getList();
-    foreach ($files as $module => $file) {
-      // Ignore disabled modules and installation profiles.
-      if (!$file->status || $module == $profile) {
+    $files += \Drupal::service('extension.list.theme')->getList();
+    $core_incompatible_extensions = [];
+    $php_incompatible_extensions = [];
+    foreach ($files as $extension_name => $file) {
+      // Ignore uninstalled extensions and installation profiles.
+      if (!$file->status || $extension_name == $profile) {
         continue;
       }
-      // Check the module's PHP version.
+
       $name = $file->info['name'];
+      if (!empty($file->info['core_incompatible'])) {
+        $core_incompatible_extensions[$file->info['type']][] = $name;
+      }
+
+      // Check the extension's PHP version.
       $php = $file->info['php'];
       if (version_compare($php, PHP_VERSION, '>')) {
-        $requirements['php']['description'] .= t('@name requires at least PHP @version.', ['@name' => $name, '@version' => $php]);
-        $requirements['php']['severity'] = REQUIREMENT_ERROR;
+        $php_incompatible_extensions[$file->info['type']][] = $name;
       }
+
+      // @todo Remove this 'if' block to allow checking requirements of themes
+      //   https://www.drupal.org/project/drupal/issues/474684.
+      if ($file->info['type'] !== 'module') {
+        continue;
+      }
+
       // Check the module's required modules.
       /** @var \Drupal\Core\Extension\Dependency $requirement */
       foreach ($file->requires as $requirement) {
         $required_module = $requirement->getName();
         // Check if the module exists.
         if (!isset($files[$required_module])) {
-          $requirements["$module-$required_module"] = [
+          $requirements["$extension_name-$required_module"] = [
             'title' => t('Unresolved dependency'),
             'description' => t('@name requires this module.', ['@name' => $name]),
             'value' => t('@required_name (Missing)', ['@required_name' => $required_module]),
@@ -868,7 +914,7 @@ function system_requirements($phase) {
         $required_name = $required_file->info['name'];
         $version = str_replace(\Drupal::CORE_COMPATIBILITY . '-', '', $required_file->info['version']);
         if (!$requirement->isCompatible($version)) {
-          $requirements["$module-$required_module"] = [
+          $requirements["$extension_name-$required_module"] = [
             'title' => t('Unresolved dependency'),
             'description' => t('@name requires this module and version. Currently using @required_name version @version', ['@name' => $name, '@required_name' => $required_name, '@version' => $version]),
             'value' => t('@required_name (Version @compatibility required)', ['@required_name' => $required_name, '@compatibility' => $requirement->getConstraintString()]),
@@ -878,6 +924,115 @@ function system_requirements($phase) {
         }
       }
     }
+    if (!empty($core_incompatible_extensions['module'])) {
+      $requirements['module_core_incompatible'] = $create_extension_incompatibility_list(
+        $core_incompatible_extensions['module'],
+        new PluralTranslatableMarkup(
+          count($core_incompatible_extensions['module']),
+        'The following module is installed, but it is incompatible with Drupal @version:',
+        'The following modules are installed, but they are incompatible with Drupal @version:',
+        ['@version' => \Drupal::VERSION]
+        ),
+        new PluralTranslatableMarkup(
+          count($core_incompatible_extensions['module']),
+          'Incompatible module',
+          'Incompatible modules'
+        )
+      );
+    }
+    if (!empty($core_incompatible_extensions['theme'])) {
+      $requirements['theme_core_incompatible'] = $create_extension_incompatibility_list(
+        $core_incompatible_extensions['theme'],
+        new PluralTranslatableMarkup(
+          count($core_incompatible_extensions['theme']),
+          'The following theme is installed, but it is incompatible with Drupal @version:',
+          'The following themes are installed, but they are incompatible with Drupal @version:',
+          ['@version' => \Drupal::VERSION]
+        ),
+        new PluralTranslatableMarkup(
+          count($core_incompatible_extensions['theme']),
+          'Incompatible theme',
+          'Incompatible themes'
+        )
+      );
+    }
+    if (!empty($php_incompatible_extensions['module'])) {
+      $requirements['module_php_incompatible'] = $create_extension_incompatibility_list(
+        $php_incompatible_extensions['module'],
+        new PluralTranslatableMarkup(
+          count($php_incompatible_extensions['module']),
+          'The following module is installed, but it is incompatible with PHP @version:',
+          'The following modules are installed, but they are incompatible with PHP @version:',
+          ['@version' => phpversion()]
+        ),
+        new PluralTranslatableMarkup(
+          count($php_incompatible_extensions['module']),
+          'Incompatible module',
+          'Incompatible modules'
+        )
+      );
+    }
+    if (!empty($php_incompatible_extensions['theme'])) {
+      $requirements['theme_php_incompatible'] = $create_extension_incompatibility_list(
+        $php_incompatible_extensions['theme'],
+        new PluralTranslatableMarkup(
+          count($php_incompatible_extensions['theme']),
+          'The following theme is installed, but it is incompatible with PHP @version:',
+          'The following themes are installed, but they are incompatible with PHP @version:',
+          ['@version' => phpversion()]
+        ),
+        new PluralTranslatableMarkup(
+          count($php_incompatible_extensions['theme']),
+          'Incompatible theme',
+          'Incompatible themes'
+        )
+      );
+    }
+
+    // Look for invalid modules.
+    $extension_config = \Drupal::configFactory()->get('core.extension');
+    /** @var \Drupal\Core\Extension\ExtensionList $extension_list */
+    $extension_list = \Drupal::service('extension.list.module');
+    $is_missing_extension = function ($extension_name) use (&$extension_list) {
+      return !$extension_list->exists($extension_name);
+    };
+
+    $invalid_modules = array_filter(array_keys($extension_config->get('module')), $is_missing_extension);
+
+    if (!empty($invalid_modules)) {
+      $requirements['invalid_module'] = $create_extension_incompatibility_list(
+        $invalid_modules,
+        new PluralTranslatableMarkup(
+          count($invalid_modules),
+          'The following module is marked as installed in the core.extension configuration, but it is missing:',
+          'The following modules are marked as installed in the core.extension configuration, but they are missing:'
+        ),
+        new PluralTranslatableMarkup(
+          count($invalid_modules),
+          'Missing or invalid module',
+          'Missing or invalid modules'
+        )
+      );
+    }
+
+    // Look for invalid themes.
+    $extension_list = \Drupal::service('extension.list.theme');
+    $invalid_themes = array_filter(array_keys($extension_config->get('theme')), $is_missing_extension);
+    if (!empty($invalid_themes)) {
+      $requirements['invalid_theme'] = $create_extension_incompatibility_list(
+        $invalid_themes,
+        new PluralTranslatableMarkup(
+          count($invalid_themes),
+          'The following theme is marked as installed in the core.extension configuration, but it is missing:',
+          'The following themes are marked as installed in the core.extension configuration, but they are missing:'
+        ),
+        new PluralTranslatableMarkup(
+          count($invalid_themes),
+          'Missing or invalid theme',
+          'Missing or invalid themes'
+        )
+      );
+    }
   }
 
   // Returns Unicode library status and errors.
diff --git a/core/modules/system/tests/src/Functional/UpdateSystem/UpdateScriptTest.php b/core/modules/system/tests/src/Functional/UpdateSystem/UpdateScriptTest.php
index 1a5cbae18fe8..e98462fdbeb6 100644
--- a/core/modules/system/tests/src/Functional/UpdateSystem/UpdateScriptTest.php
+++ b/core/modules/system/tests/src/Functional/UpdateSystem/UpdateScriptTest.php
@@ -2,6 +2,7 @@
 
 namespace Drupal\Tests\system\Functional\UpdateSystem;
 
+use Drupal\Component\Serialization\Yaml;
 use Drupal\Core\Url;
 use Drupal\language\Entity\ConfigurableLanguage;
 use Drupal\Tests\BrowserTestBase;
@@ -16,6 +17,8 @@ class UpdateScriptTest extends BrowserTestBase {
 
   use RequirementsPageTrait;
 
+  protected const HANDBOOK_MESSAGE = 'Review the suggestions for resolving this incompatibility to repair your installation, and then re-run update.php.';
+
   /**
    * Modules to enable.
    *
@@ -33,6 +36,13 @@ class UpdateScriptTest extends BrowserTestBase {
    */
   protected $dumpHeaders = TRUE;
 
+  /**
+   * The URL to the status report page.
+   *
+   * @var \Drupal\Core\Url
+   */
+  protected $statusReportUrl;
+
   /**
    * URL to the update.php script.
    *
@@ -50,6 +60,7 @@ class UpdateScriptTest extends BrowserTestBase {
   protected function setUp() {
     parent::setUp();
     $this->updateUrl = Url::fromRoute('system.db_update');
+    $this->statusReportUrl = Url::fromRoute('system.status');
     $this->updateUser = $this->drupalCreateUser(['administer software updates', 'access site in maintenance mode']);
   }
 
@@ -166,6 +177,227 @@ public function testRequirements() {
     $this->assertSession()->responseContains('Update script test requires this module and version. Currently using Node version ' . \Drupal::VERSION);
   }
 
+  /**
+   * Tests that extension compatibility changes are handled correctly.
+   *
+   * @param array $correct_info
+   *   The initial values for info.yml fail. These should compatible with core.
+   * @param array $breaking_info
+   *   The values to the info.yml that are not compatible with core.
+   * @param string $expected_error
+   *   The expected error.
+   *
+   * @dataProvider providerExtensionCompatibilityChange
+   */
+  public function testExtensionCompatibilityChange(array $correct_info, array $breaking_info, $expected_error) {
+    $extension_type = $correct_info['type'];
+    $this->drupalLogin(
+      $this->drupalCreateUser(
+        [
+          'administer software updates',
+          'administer site configuration',
+          $extension_type === 'module' ? 'administer modules' : 'administer themes',
+        ]
+      )
+    );
+
+    $extension_machine_name = "changing_extension";
+    $extension_name = "$extension_machine_name name";
+
+    $test_error_text = "Incompatible $extension_type "
+      . $expected_error
+      . $extension_name
+      . static::HANDBOOK_MESSAGE;
+    $base_info = ['name' => $extension_name];
+    if ($extension_type === 'theme') {
+      $base_info['base theme'] = FALSE;
+    }
+    $folder_path = \Drupal::service('site.path') . "/{$extension_type}s/$extension_machine_name";
+    $file_path = "$folder_path/$extension_machine_name.info.yml";
+    mkdir($folder_path, 0777, TRUE);
+    file_put_contents($file_path, Yaml::encode($base_info + $correct_info));
+    $this->enableExtension($extension_type, $extension_machine_name, $extension_name);
+    $this->assertInstalledExtensionConfig($extension_type, $extension_machine_name);
+
+    // If there are no requirements warnings or errors, we expect to be able to
+    // go through the update process uninterrupted.
+    $this->assertUpdateWithNoError($test_error_text, $extension_type, $extension_machine_name);
+
+    // Change the values in the info.yml and confirm updating is not possible.
+    file_put_contents($file_path, Yaml::encode($base_info + $breaking_info));
+    $this->assertErrorOnUpdate($test_error_text, $extension_type, $extension_machine_name);
+
+    // Fix the values in the info.yml file and confirm updating is possible
+    // again.
+    file_put_contents($file_path, Yaml::encode($base_info + $correct_info));
+    $this->assertUpdateWithNoError($test_error_text, $extension_type, $extension_machine_name);
+  }
+
+  /**
+   * Date provider for testExtensionCompatibilityChange().
+   */
+  public function providerExtensionCompatibilityChange() {
+    $incompatible_module_message = "The following module is installed, but it is incompatible with Drupal " . \Drupal::VERSION . ":";
+    $incompatible_theme_message = "The following theme is installed, but it is incompatible with Drupal " . \Drupal::VERSION . ":";
+    return [
+      'module: core key incompatible' => [
+        [
+          'core_version_requirement' => '^8 || ^9',
+          'type' => 'module',
+        ],
+        [
+          'core' => '7.x',
+          'type' => 'module',
+        ],
+        $incompatible_module_message,
+      ],
+      'module: core_version_requirement key incompatible' => [
+        [
+          'core_version_requirement' => '^8 || ^9',
+          'type' => 'module',
+        ],
+        [
+          'core_version_requirement' => '8.7.7',
+          'type' => 'module',
+        ],
+        $incompatible_module_message,
+      ],
+      'theme: core key incompatible' => [
+        [
+          'core_version_requirement' => '^8 || ^9',
+          'type' => 'theme',
+        ],
+        [
+          'core' => '7.x',
+          'type' => 'theme',
+        ],
+        $incompatible_theme_message,
+      ],
+      'theme: core_version_requirement key incompatible' => [
+        [
+          'core_version_requirement' => '^8 || ^9',
+          'type' => 'theme',
+        ],
+        [
+          'core_version_requirement' => '8.7.7',
+          'type' => 'theme',
+        ],
+        $incompatible_theme_message,
+      ],
+      'module: php requirement' => [
+        [
+          'core_version_requirement' => '^8 || ^9',
+          'type' => 'module',
+          'php' => 1,
+        ],
+        [
+          'core_version_requirement' => '^8 || ^9',
+          'type' => 'module',
+          'php' => 1000000000,
+        ],
+        'The following module is installed, but it is incompatible with PHP ' . phpversion() . ":",
+      ],
+      'theme: php requirement' => [
+        [
+          'core_version_requirement' => '^8 || ^9',
+          'type' => 'theme',
+          'php' => 1,
+        ],
+        [
+          'core_version_requirement' => '^8 || ^9',
+          'type' => 'theme',
+          'php' => 1000000000,
+        ],
+        'The following theme is installed, but it is incompatible with PHP ' . phpversion() . ":",
+      ],
+    ];
+  }
+
+  /**
+   * Tests that a missing extension prevents updates.
+   *
+   * @param string $extension_type
+   *   The extension type, either 'module' or 'theme'.
+   *
+   * @dataProvider providerMissingExtension
+   */
+  public function testMissingExtension($extension_type) {
+    $this->drupalLogin(
+      $this->drupalCreateUser(
+        [
+          'administer software updates',
+          'administer site configuration',
+          $extension_type === 'module' ? 'administer modules' : 'administer themes',
+        ]
+      )
+    );
+    $extension_machine_name = "disappearing_$extension_type";
+    $extension_name = 'The magically disappearing extension';
+    $test_error_text = "Missing or invalid $extension_type "
+      . "The following $extension_type is marked as installed in the core.extension configuration, but it is missing:"
+      . $extension_machine_name
+      . static::HANDBOOK_MESSAGE;
+    $extension_info = [
+      'name' => $extension_name,
+      'type' => $extension_type,
+      'core_version_requirement' => '^8 || ^9',
+    ];
+    if ($extension_type === 'theme') {
+      $extension_info['base theme'] = FALSE;
+    }
+    $folder_path = \Drupal::service('site.path') . "/{$extension_type}s/$extension_machine_name";
+    $file_path = "$folder_path/$extension_machine_name.info.yml";
+    mkdir($folder_path, 0777, TRUE);
+    file_put_contents($file_path, Yaml::encode($extension_info));
+    $this->enableExtension($extension_type, $extension_machine_name, $extension_name);
+
+    // If there are no requirements warnings or errors, we expect to be able to
+    // go through the update process uninterrupted.
+    $this->assertUpdateWithNoError($test_error_text, $extension_type, $extension_machine_name);
+
+    // Delete the info.yml and confirm updates are prevented.
+    unlink($file_path);
+    $this->assertErrorOnUpdate($test_error_text, $extension_type, $extension_machine_name);
+
+    // Add the info.yml file back and confirm we are able to go through the
+    // update process uninterrupted.
+    file_put_contents($file_path, Yaml::encode($extension_info));
+    $this->assertUpdateWithNoError($test_error_text, $extension_type, $extension_machine_name);
+  }
+
+  /**
+   * Data provider for testMissingExtension().
+   */
+  public function providerMissingExtension() {
+    return [
+      'module' => ['module'],
+      'theme' => ['theme'],
+    ];
+  }
+
+  /**
+   * Enables an extension using the UI.
+   *
+   * @param string $extension_type
+   *   The extension type.
+   * @param string $extension_machine_name
+   *   The extension machine name.
+   * @param string $extension_name
+   *   The extension name.
+   */
+  protected function enableExtension($extension_type, $extension_machine_name, $extension_name) {
+    if ($extension_type === 'module') {
+      $edit = [
+        "modules[$extension_machine_name][enable]" => $extension_machine_name,
+      ];
+      $this->drupalPostForm('admin/modules', $edit, t('Install'));
+    }
+    elseif ($extension_type === 'theme') {
+      $this->drupalGet('admin/appearance');
+      $this->click("a[title~=\"$extension_name\"]");
+    }
+  }
+
   /**
    * Tests the effect of using the update script on the theme system.
    */
@@ -431,4 +663,69 @@ public function getSystemSchema() {
     ];
   }
 
+  /**
+   * Asserts that an installed extension's config setting is correct.
+   *
+   * @param string $extension_type
+   *   The extension type, either 'module' or 'theme'.
+   * @param string $extension_machine_name
+   *   The extension machine name.
+   */
+  protected function assertInstalledExtensionConfig($extension_type, $extension_machine_name) {
+    $extension_config = $this->container->get('config.factory')->getEditable('core.extension');
+    $this->assertSame(0, $extension_config->get("$extension_type.$extension_machine_name"));
+  }
+
+  /**
+   * Asserts a particular error is not shown on update and status report pages.
+   *
+   * @param string $unexpected_error_text
+   *   The error text that should not be shown.
+   * @param string $extension_type
+   *   The extension type, either 'module' or 'theme'.
+   * @param string $extension_machine_name
+   *   The extension machine name.
+   *
+   * @throws \Behat\Mink\Exception\ResponseTextException
+   */
+  protected function assertUpdateWithNoError($unexpected_error_text, $extension_type, $extension_machine_name) {
+    $assert_session = $this->assertSession();
+    $this->drupalGet($this->statusReportUrl);
+    $this->assertSession()->pageTextNotContains($unexpected_error_text);
+    $this->drupalGet($this->updateUrl, ['external' => TRUE]);
+    $this->assertSession()->pageTextNotContains($unexpected_error_text);
+    $this->updateRequirementsProblem();
+    $this->clickLink(t('Continue'));
+    $assert_session->pageTextContains('No pending updates.');
+    $this->assertInstalledExtensionConfig($extension_type, $extension_machine_name);
+  }
+
+  /**
+   * Asserts an error is shown on the update and status report pages.
+   *
+   * @param string $expected_error_text
+   *   The expected error text.
+   * @param string $extension_type
+   *   The extension type, either 'module' or 'theme'.
+   * @param string $extension_machine_name
+   *   The extension machine name.
+   *
+   * @throws \Behat\Mink\Exception\ExpectationException
+   * @throws \Behat\Mink\Exception\ResponseTextException
+   */
+  protected function assertErrorOnUpdate($expected_error_text, $extension_type, $extension_machine_name) {
+    $assert_session = $this->assertSession();
+    $this->drupalGet($this->statusReportUrl);
+    $this->assertSession()->pageTextContains($expected_error_text);
+
+    // Reload the update page to ensure the extension with the breaking values
+    // has not been uninstalled or otherwise affected.
+    for ($reload = 0; $reload <= 1; $reload++) {
+      $this->drupalGet($this->updateUrl, ['external' => TRUE]);
+      $this->assertSession()->pageTextContains($expected_error_text);
+      $assert_session->linkNotExists('Continue');
+    }
+    $this->assertInstalledExtensionConfig($extension_type, $extension_machine_name);
+  }
+
 }
diff --git a/core/tests/Drupal/KernelTests/Core/Update/CompatibilityFixTest.php b/core/tests/Drupal/KernelTests/Core/Update/CompatibilityFixTest.php
index 20a45cad5c1a..6fcae686bf94 100644
--- a/core/tests/Drupal/KernelTests/Core/Update/CompatibilityFixTest.php
+++ b/core/tests/Drupal/KernelTests/Core/Update/CompatibilityFixTest.php
@@ -8,6 +8,7 @@
  * Tests that extensions that are incompatible with the current core version are disabled.
  *
  * @group Update
+ * @group legacy
  */
 class CompatibilityFixTest extends KernelTestBase {
 
@@ -21,6 +22,9 @@ protected function setUp() {
     require_once $this->root . '/core/includes/update.inc';
   }
 
+  /**
+   * @expectedDeprecation update_fix_compatibility() is deprecated in Drupal 8.8.4 and will be removed before Drupal 9.0.0. There is no replacement. See https://www.drupal.org/node/3026100
+   */
   public function testFixCompatibility() {
     $extension_config = \Drupal::configFactory()->getEditable('core.extension');
 
diff --git a/core/tests/Drupal/Tests/Core/Extension/ExtensionListTest.php b/core/tests/Drupal/Tests/Core/Extension/ExtensionListTest.php
index ad188a4ed34b..8da6fa436026 100644
--- a/core/tests/Drupal/Tests/Core/Extension/ExtensionListTest.php
+++ b/core/tests/Drupal/Tests/Core/Extension/ExtensionListTest.php
@@ -206,9 +206,77 @@ public function testReset() {
   }
 
   /**
+   * @covers ::checkIncompatibility
+   *
+   * @dataProvider providerCheckIncompatibility
+   */
+  public function testCheckIncompatibility($additional_settings, $expected) {
+    $test_extension_list = $this->setupTestExtensionList(['test_name'], $additional_settings);
+    $this->assertSame($expected, $test_extension_list->checkIncompatibility('test_name'));
+  }
+
+  /**
+   * DataProvider for testCheckIncompatibility().
+   */
+  public function providerCheckIncompatibility() {
+    return [
+      'core_incompatible true' => [
+        [
+          'core_incompatible' => TRUE,
+        ],
+        TRUE,
+      ],
+      'core_incompatible false' => [
+        [
+          'core_incompatible' => FALSE,
+        ],
+        FALSE,
+      ],
+      'PHP 1, core_incompatible FALSE' => [
+        [
+          'core_incompatible' => FALSE,
+          'php' => 1,
+        ],
+        FALSE,
+      ],
+      'PHP 1000000000000, core_incompatible FALSE' => [
+        [
+          'core_incompatible' => FALSE,
+          'php' => 1000000000000,
+        ],
+        TRUE,
+      ],
+      'PHP 1, core_incompatible TRUE' => [
+        [
+          'core_incompatible' => TRUE,
+          'php' => 1,
+        ],
+        TRUE,
+      ],
+      'PHP 1000000000000, core_incompatible TRUE' => [
+        [
+          'core_incompatible' => TRUE,
+          'php' => 1000000000000,
+        ],
+        TRUE,
+      ],
+    ];
+  }
+
+  /**
+   * Sets up an a test extension list.
+   *
+   * @param string[] $extension_names
+   *   The names of the extensions to create.
+   * @param mixed[] $additional_info_values
+   *   The additional values to add to extensions info.yml files. These values
+   *   will be encoded using '\Drupal\Component\Serialization\Yaml::encode()'.
+   *   The array keys should be valid top level yaml file keys.
+   *
    * @return \Drupal\Tests\Core\Extension\TestExtension
+   *   The test extension list.
    */
-  protected function setupTestExtensionList($extension_names = ['test_name']) {
+  protected function setupTestExtensionList(array $extension_names = ['test_name'], array $additional_info_values = []) {
     vfsStream::setup('drupal_root');
 
     $folders = ['example' => []];
@@ -217,7 +285,7 @@ protected function setupTestExtensionList($extension_names = ['test_name']) {
         'name' => 'test name',
         'type' => 'test_extension',
         'core' => '8.x',
-      ]);
+      ] + $additional_info_values);
     }
     vfsStream::create($folders);
     foreach ($extension_names as $extension_name) {
-- 
GitLab