From 6e84b5797c2693b63fc32ffc08225c40b0e4ce83 Mon Sep 17 00:00:00 2001 From: catch <catch@35733.no-reply.drupal.org> Date: Thu, 1 Apr 2021 19:56:13 +0100 Subject: [PATCH] Issue #2160091 by alexpott, dawehner, damiankloip, swentel, sun, moshe weitzman, xjm, joelpittet, charginghawk: drupal_rebuild() rebuilds container twice, since drupal_flush_all_caches() also rebuilds it --- core/includes/common.inc | 56 +++++++++--------- core/includes/utility.inc | 16 +----- .../container_rebuild_test.info.yml | 5 ++ .../container_rebuild_test.routing.yml | 6 ++ .../ContainerRebuildTestServiceProvider.php | 18 ++++++ .../src/TestController.php | 37 ++++++++++++ .../UpdateSystem/RebuildScriptTest.php | 57 +++++++++++++++++++ .../Core/Common/DrupalFlushAllCachesTest.php | 26 ++++++++- 8 files changed, 177 insertions(+), 44 deletions(-) create mode 100644 core/modules/system/tests/modules/container_rebuild_test/container_rebuild_test.info.yml create mode 100644 core/modules/system/tests/modules/container_rebuild_test/container_rebuild_test.routing.yml create mode 100644 core/modules/system/tests/modules/container_rebuild_test/src/ContainerRebuildTestServiceProvider.php create mode 100644 core/modules/system/tests/modules/container_rebuild_test/src/TestController.php diff --git a/core/includes/common.inc b/core/includes/common.inc index 42d9b80d2827..f640a901219e 100644 --- a/core/includes/common.inc +++ b/core/includes/common.inc @@ -12,6 +12,7 @@ use Drupal\Component\Utility\Html; use Drupal\Component\Utility\SortArray; use Drupal\Core\Cache\Cache; +use Drupal\Core\DrupalKernel; use Drupal\Core\StringTranslation\TranslatableMarkup; /** @@ -460,10 +461,11 @@ function show(&$element) { } /** - * Flushes all persistent caches, resets all variables, and rebuilds all data structures. + * Rebuilds the container, flushes all persistent caches, resets all variables, and rebuilds all data structures. * * At times, it is necessary to re-initialize the entire system to account for * changed or new code. This function: + * - Rebuilds the container if $kernel is not passed in. * - Clears all persistent caches: * - The bootstrap cache bin containing base system, module system, and theme * system information. @@ -483,6 +485,10 @@ function show(&$element) { * - The 'active' status of fields is refreshed. * - Rebuilds the menu router. * + * It's discouraged to call this during a regular page request. + * If you call this function in tests, every code afterwards should use the new + * container. + * * This means the entire system is reset so all caches and static variables are * effectively empty. After that is guaranteed, information about the currently * active code is updated, and rebuild operations are successively called in @@ -512,12 +518,19 @@ function show(&$element) { * cache though.) * @todo Add a global lock to ensure that caches are not primed in concurrent * requests. + * + * @param \Drupal\Core\DrupalKernel|array $kernel + * (optional) The Drupal Kernel. It is the caller's responsibility to rebuild + * the container if this is passed in. Sometimes drupal_flush_all_caches is + * used as a batch operation so $kernel will be an array, in this instance it + * will be treated as if it it NULL. */ -function drupal_flush_all_caches() { +function drupal_flush_all_caches($kernel = NULL) { + // This is executed based on old/previously known information if $kernel is + // not passed in, which is sufficient, since new extensions cannot have any + // primed caches yet. $module_handler = \Drupal::moduleHandler(); // Flush all persistent caches. - // This is executed based on old/previously known information, which is - // sufficient, since new extensions cannot have any primed caches yet. $module_handler->invokeAll('cache_flush'); foreach (Cache::getBins() as $service_id => $cache_backend) { $cache_backend->deleteAll(); @@ -531,43 +544,26 @@ function drupal_flush_all_caches() { // Reset all static caches. drupal_static_reset(); - // Invalidate the container. - \Drupal::service('kernel')->invalidateContainer(); - // Wipe the Twig PHP Storage cache. \Drupal::service('twig')->invalidate(); - // Rebuild module and theme data. - $module_data = \Drupal::service('extension.list.module')->reset()->getList(); - /** @var \Drupal\Core\Extension\ThemeHandlerInterface $theme_handler */ - $theme_handler = \Drupal::service('theme_handler'); - $theme_handler->refreshInfo(); + // Rebuild theme data that is stored in state. + \Drupal::service('theme_handler')->refreshInfo(); // In case the active theme gets requested later in the same request we need // to reset the theme manager. \Drupal::theme()->resetActiveTheme(); - // Rebuild and reboot a new kernel. A simple DrupalKernel reboot is not - // sufficient, since the list of enabled modules might have been adjusted - // above due to changed code. - $files = []; - $modules = []; - foreach ($module_data as $name => $extension) { - if ($extension->status) { - $files[$name] = $extension; - $modules[$name] = $extension->weight; - } + if (!$kernel instanceof DrupalKernel) { + $kernel = \Drupal::service('kernel'); + $kernel->invalidateContainer(); + $kernel->rebuildContainer(); } - $modules = module_config_sort($modules); - \Drupal::service('kernel')->updateModules($modules, $files); - // New container, new module handler. - $module_handler = \Drupal::moduleHandler(); - // Ensure that all modules that are currently supposed to be enabled are - // actually loaded. - $module_handler->loadAll(); + // Rebuild module data that is stored in state. + \Drupal::service('extension.list.module')->reset(); // Rebuild all information based on new module data. - $module_handler->invokeAll('rebuild'); + \Drupal::moduleHandler()->invokeAll('rebuild'); // Clear all plugin caches. \Drupal::service('plugin.cache_clearer')->clearCachedDefinitions(); diff --git a/core/includes/utility.inc b/core/includes/utility.inc index 3a6ef0bde289..13975bd18bf3 100644 --- a/core/includes/utility.inc +++ b/core/includes/utility.inc @@ -5,8 +5,6 @@ * Miscellaneous functions. */ -use Drupal\Core\PhpStorage\PhpStorageFactory; -use Drupal\Core\Cache\Cache; use Drupal\Core\DrupalKernel; use Symfony\Component\HttpFoundation\Request; @@ -28,12 +26,11 @@ function drupal_rebuild($class_loader, Request $request) { restore_error_handler(); restore_exception_handler(); - // Force kernel to rebuild php cache. - PhpStorageFactory::get('twig')->deleteAll(); - + // Invalidate the container. // Bootstrap up to where caches exist and clear them. $kernel = new DrupalKernel('prod', $class_loader); $kernel->setSitePath(DrupalKernel::findSitePath($request)); + $kernel->invalidateContainer(); $kernel->boot(); $kernel->preHandle($request); // Ensure our request includes the session if appropriate. @@ -41,18 +38,11 @@ function drupal_rebuild($class_loader, Request $request) { $request->setSession($kernel->getContainer()->get('session')); } - // Invalidate the container. - $kernel->invalidateContainer(); - - foreach (Cache::getBins() as $bin) { - $bin->deleteAll(); - } + drupal_flush_all_caches($kernel); // Disable recording of cached pages. \Drupal::service('page_cache_kill_switch')->trigger(); - drupal_flush_all_caches(); - // Restore Drupal's error and exception handlers. // @see \Drupal\Core\DrupalKernel::boot() set_error_handler('_drupal_error_handler'); diff --git a/core/modules/system/tests/modules/container_rebuild_test/container_rebuild_test.info.yml b/core/modules/system/tests/modules/container_rebuild_test/container_rebuild_test.info.yml new file mode 100644 index 000000000000..2c82600fd630 --- /dev/null +++ b/core/modules/system/tests/modules/container_rebuild_test/container_rebuild_test.info.yml @@ -0,0 +1,5 @@ +name: 'Service Provider test' +type: module +description: 'Support module for service provider testing.' +package: Testing +version: VERSION diff --git a/core/modules/system/tests/modules/container_rebuild_test/container_rebuild_test.routing.yml b/core/modules/system/tests/modules/container_rebuild_test/container_rebuild_test.routing.yml new file mode 100644 index 000000000000..0382c21dd25c --- /dev/null +++ b/core/modules/system/tests/modules/container_rebuild_test/container_rebuild_test.routing.yml @@ -0,0 +1,6 @@ +container_rebuild_test.module_path: + path: '/container_rebuild_test/{module}/{function}' + defaults: + _controller: '\Drupal\container_rebuild_test\TestController::showModuleInfo' + requirements: + _access: 'TRUE' diff --git a/core/modules/system/tests/modules/container_rebuild_test/src/ContainerRebuildTestServiceProvider.php b/core/modules/system/tests/modules/container_rebuild_test/src/ContainerRebuildTestServiceProvider.php new file mode 100644 index 000000000000..d5f2bb808cb6 --- /dev/null +++ b/core/modules/system/tests/modules/container_rebuild_test/src/ContainerRebuildTestServiceProvider.php @@ -0,0 +1,18 @@ +<?php + +namespace Drupal\container_rebuild_test; + +use Drupal\Core\DependencyInjection\ContainerBuilder; +use Drupal\Core\DependencyInjection\ServiceModifierInterface; + +class ContainerRebuildTestServiceProvider implements ServiceModifierInterface { + + /** + * {@inheritdoc} + */ + public function alter(ContainerBuilder $container) { + $count = $container->get('state')->get('container_rebuild_test.count', 0); + $container->get('state')->set('container_rebuild_test.count', ++$count); + } + +} diff --git a/core/modules/system/tests/modules/container_rebuild_test/src/TestController.php b/core/modules/system/tests/modules/container_rebuild_test/src/TestController.php new file mode 100644 index 000000000000..7fc722faccc7 --- /dev/null +++ b/core/modules/system/tests/modules/container_rebuild_test/src/TestController.php @@ -0,0 +1,37 @@ +<?php + +namespace Drupal\container_rebuild_test; + +use Drupal\Core\Controller\ControllerBase; + +class TestController extends ControllerBase { + + /** + * Displays the path to a module. + * + * @param string $module + * The module name. + * @param string $function + * The function to check if it exists. + * + * @return string[] + * A render array. + */ + public function showModuleInfo(string $module, string $function) { + $module_handler = \Drupal::moduleHandler(); + $module_message = $module . ': '; + if ($module_handler->moduleExists($module)) { + $module_message .= \Drupal::moduleHandler()->getModule($module)->getPath(); + } + else { + $module_message .= 'not installed'; + } + $function_message = $function . ': ' . var_export(function_exists($function), TRUE); + + return [ + '#theme' => 'item_list', + '#items' => [$module_message, $function_message], + ]; + } + +} diff --git a/core/modules/system/tests/src/Functional/UpdateSystem/RebuildScriptTest.php b/core/modules/system/tests/src/Functional/UpdateSystem/RebuildScriptTest.php index 10888a471f1e..e5e0baf6c192 100644 --- a/core/modules/system/tests/src/Functional/UpdateSystem/RebuildScriptTest.php +++ b/core/modules/system/tests/src/Functional/UpdateSystem/RebuildScriptTest.php @@ -4,6 +4,7 @@ use Drupal\Core\Url; use Drupal\Tests\BrowserTestBase; +use Symfony\Component\Filesystem\Filesystem; /** * Tests the rebuild script access and functionality. @@ -12,6 +13,11 @@ */ class RebuildScriptTest extends BrowserTestBase { + /** + * {@inheritdoc} + */ + protected static $modules = ['module_test', 'container_rebuild_test']; + /** * {@inheritdoc} */ @@ -37,9 +43,60 @@ public function testRebuild() { $this->rebuildAll(); $cache->set('rebuild_test', TRUE); + \Drupal::state()->set('container_rebuild_test.count', 0); + $this->drupalGet(Url::fromUri('base:core/rebuild.php')); + $this->assertSession()->addressEquals(new Url('<front>')); + $this->assertFalse($cache->get('rebuild_test')); + $this->refreshVariables(); + $this->assertSame(1, \Drupal::state()->get('container_rebuild_test.count', 0)); + $this->drupalGet('/container_rebuild_test/module_test/module_test_system_info_alter'); + $this->assertSession()->pageTextContains('module_test: core/modules/system/tests/modules/module_test'); + $this->assertSession()->pageTextContains('module_test_system_info_alter: true'); + + // Move a module to ensure it does not break the rebuild. + $file_system = new Filesystem(); + $file_system->mirror('core/modules/system/tests/modules/module_test', $this->siteDirectory . '/modules/module_test'); + \Drupal::state()->set('container_rebuild_test.count', 0); + $this->drupalGet(Url::fromUri('base:core/rebuild.php')); + $this->assertSession()->addressEquals(new Url('<front>')); + $this->refreshVariables(); + $this->assertSame(1, \Drupal::state()->get('container_rebuild_test.count', 0)); + $this->drupalGet('/container_rebuild_test/module_test/module_test_system_info_alter'); + $this->assertSession()->pageTextContains('module_test: ' . $this->siteDirectory . '/modules/module_test'); + $this->assertSession()->pageTextContains('module_test_system_info_alter: true'); + + // Disable a module by writing to the core.extension list. + $this->config('core.extension')->clear('module.module_test')->save(); + \Drupal::state()->set('container_rebuild_test.count', 0); + $this->drupalGet(Url::fromUri('base:core/rebuild.php')); + $this->assertSession()->addressEquals(new Url('<front>')); + $this->refreshVariables(); + $this->assertSame(1, \Drupal::state()->get('container_rebuild_test.count', 0)); + $this->drupalGet('/container_rebuild_test/module_test/module_test_system_info_alter'); + $this->assertSession()->pageTextContains('module_test: not installed'); + $this->assertSession()->pageTextContains('module_test_system_info_alter: false'); + + // Enable a module by writing to the core.extension list. + $modules = $this->config('core.extension')->get('module'); + $modules['module_test'] = 0; + $this->config('core.extension')->set('module', module_config_sort($modules))->save(); + \Drupal::state()->set('container_rebuild_test.count', 0); + $this->drupalGet(Url::fromUri('base:core/rebuild.php')); + $this->assertSession()->addressEquals(new Url('<front>')); + $this->refreshVariables(); + $this->assertSame(1, \Drupal::state()->get('container_rebuild_test.count', 0)); + $this->drupalGet('/container_rebuild_test/module_test/module_test_system_info_alter'); + $this->assertSession()->pageTextContains('module_test: ' . $this->siteDirectory . '/modules/module_test'); + $this->assertSession()->pageTextContains('module_test_system_info_alter: true'); + + // Test how many container rebuild occur when there is no cached container. + \Drupal::state()->set('container_rebuild_test.count', 0); + \Drupal::service('kernel')->invalidateContainer(); $this->drupalGet(Url::fromUri('base:core/rebuild.php')); $this->assertSession()->addressEquals(new Url('<front>')); $this->assertFalse($cache->get('rebuild_test')); + $this->refreshVariables(); + $this->assertSame(1, \Drupal::state()->get('container_rebuild_test.count', 0)); } } diff --git a/core/tests/Drupal/KernelTests/Core/Common/DrupalFlushAllCachesTest.php b/core/tests/Drupal/KernelTests/Core/Common/DrupalFlushAllCachesTest.php index 0953c05a4c7c..9cb1dc2defc7 100644 --- a/core/tests/Drupal/KernelTests/Core/Common/DrupalFlushAllCachesTest.php +++ b/core/tests/Drupal/KernelTests/Core/Common/DrupalFlushAllCachesTest.php @@ -2,6 +2,7 @@ namespace Drupal\KernelTests\Core\Common; +use Drupal\Core\DependencyInjection\ContainerBuilder; use Drupal\KernelTests\KernelTestBase; /** @@ -10,6 +11,13 @@ */ class DrupalFlushAllCachesTest extends KernelTestBase { + /** + * Stores the number of container builds. + * + * @var int + */ + protected $containerBuilds = 0; + /** * {@inheritdoc} */ @@ -19,14 +27,30 @@ class DrupalFlushAllCachesTest extends KernelTestBase { * Tests that drupal_flush_all_caches() uses core.extension properly. */ public function testDrupalFlushAllCachesModuleList() { + $this->assertFalse(function_exists('system_test_help')); $core_extension = \Drupal::configFactory()->getEditable('core.extension'); $module = $core_extension->get('module'); $module['system_test'] = -10; $core_extension->set('module', module_config_sort($module))->save(); + $this->containerBuilds = 0; + drupal_flush_all_caches(); + $this->assertSame(['system_test', 'system'], array_keys($this->container->getParameter('container.modules'))); + $this->assertSame(1, $this->containerBuilds); + $this->assertTrue(function_exists('system_test_help')); + $core_extension->clear('module.system_test')->save(); + $this->containerBuilds = 0; drupal_flush_all_caches(); + $this->assertSame(['system'], array_keys($this->container->getParameter('container.modules'))); + $this->assertSame(1, $this->containerBuilds); + } - $this->assertSame(['system_test', 'system'], array_keys($this->container->getParameter('container.modules'))); + /** + * {@inheritdoc} + */ + public function register(ContainerBuilder $container) { + parent::register($container); + $this->containerBuilds++; } } -- GitLab