diff --git a/src/CronUpdater.php b/src/CronUpdater.php index 3ca7530d64e0cafe7fe9e549dcce78e46d1a2796..c3fb2eec18ba5ccbc7fbdc294f21baf46d64cf97 100644 --- a/src/CronUpdater.php +++ b/src/CronUpdater.php @@ -161,6 +161,28 @@ class CronUpdater extends Updater { private function performUpdate(string $target_version, ?int $timeout): void { $project_info = new ProjectInfo('drupal'); + if (!$this->isAvailable()) { + if ($project_info->isInstalledVersionSafe() && !$this->isApplying()) { + $this->logger->notice('Cron will not perform any updates because there is an existing stage and the current version of the site is secure.'); + return; + } + if (!$project_info->isInstalledVersionSafe() && $this->isApplying()) { + $this->logger->notice( + 'Cron will not perform any updates as an existing staged update is applying. The site is currently on an insecure version of Drupal core but will attempt to update to a secure version next time cron is run. This update may be applied manually at the <a href="%url">update form</a>.', + ['%url' => Url::fromRoute('update.report_update')->setAbsolute()->toString()], + ); + return; + } + } + + // Delete the existing staging area if not available and the site is + // currently on an insecure version. + if (!$project_info->isInstalledVersionSafe() && !$this->isAvailable() && !$this->isApplying()) { + // @todo Improve this in https://www.drupal.org/i/3325654. + $this->logger->notice('The existing stage was not in the process of being applied, so it was destroyed to allow updating the site to a secure version during cron.'); + $this->destroy(TRUE); + } + $installed_version = $project_info->getInstalledVersion(); if (empty($installed_version)) { $this->logger->error('Unable to determine the current version of Drupal core.'); diff --git a/tests/src/Build/CoreUpdateTest.php b/tests/src/Build/CoreUpdateTest.php index 6c1cede81f461093794b0babb832dcc30d2590d7..10296accb1f27d760fb460cfc90b0f8cc0bf5357 100644 --- a/tests/src/Build/CoreUpdateTest.php +++ b/tests/src/Build/CoreUpdateTest.php @@ -4,6 +4,7 @@ declare(strict_types = 1); namespace Drupal\Tests\automatic_updates\Build; +use Behat\Mink\Element\DocumentElement; use Drupal\automatic_updates\CronUpdater; use Drupal\automatic_updates\Updater; use Drupal\Composer\Composer; @@ -146,20 +147,7 @@ class CoreUpdateTest extends UpdateTestBase { $session = $mink->getSession(); $page = $session->getPage(); $assert_session = $mink->assertSession(); - - $this->visit('/admin/modules'); - $assert_session->pageTextContains('There is a security update available for your version of Drupal.'); - $page->clickLink('Update'); - - // Ensure that the update is prevented if the web root and/or vendor - // directories are not writable. - $this->assertReadOnlyFileSystemError(parse_url($session->getCurrentUrl(), PHP_URL_PATH)); - $session->reload(); - - $assert_session->pageTextNotContains('There is a security update available for your version of Drupal.'); - $page->pressButton('Update to 9.8.1'); - $this->waitForBatchJob(); - $assert_session->pageTextContains('Ready to update'); + $this->coreUpdateTillUpdateReady($page); $page->pressButton('Continue'); $this->waitForBatchJob(); $assert_session->pageTextContains('Update complete!'); @@ -230,6 +218,22 @@ class CoreUpdateTest extends UpdateTestBase { $this->webAssert->statusMessageNotExists('warning'); } + /** + * Tests stage is destroyed if not available and site is on insecure version. + */ + public function testStageDestroyedIfNotAvailable(): void { + $this->createTestProject('RecommendedProject'); + $mink = $this->getMink(); + $session = $mink->getSession(); + $page = $session->getPage(); + $assert_session = $mink->assertSession(); + $this->coreUpdateTillUpdateReady($page); + $this->visit('/admin/reports/status'); + $assert_session->pageTextContains('Your site is ready for automatic updates.'); + $page->clickLink('Run cron'); + $this->assertUpdateSuccessful('9.8.1'); + } + /** * Asserts that the update is prevented if the filesystem isn't writable. * @@ -373,4 +377,28 @@ class CoreUpdateTest extends UpdateTestBase { $this->assertCoreVersion($expected_version); } + /** + * Performs core update till update ready form. + * + * @param \Behat\Mink\Element\DocumentElement $page + * The page element. + */ + private function coreUpdateTillUpdateReady(DocumentElement $page): void { + $session = $this->getMink()->getSession(); + $this->visit('/admin/modules'); + $assert_session = $this->getMink()->assertSession($session); + $assert_session->pageTextContains('There is a security update available for your version of Drupal.'); + $page->clickLink('Update'); + + // Ensure that the update is prevented if the web root and/or vendor + // directories are not writable. + $this->assertReadOnlyFileSystemError(parse_url($session->getCurrentUrl(), PHP_URL_PATH)); + $session->reload(); + + $assert_session->pageTextNotContains('There is a security update available for your version of Drupal.'); + $page->pressButton('Update to 9.8.1'); + $this->waitForBatchJob(); + $assert_session->pageTextContains('Ready to update'); + } + } diff --git a/tests/src/Kernel/CronUpdaterTest.php b/tests/src/Kernel/CronUpdaterTest.php index 7756ec147bbbaa7167fdc9cbb1cebbb16da2b388..009b0ac78fee6217a203f3adf61e569d0869e946 100644 --- a/tests/src/Kernel/CronUpdaterTest.php +++ b/tests/src/Kernel/CronUpdaterTest.php @@ -22,6 +22,7 @@ use Drupal\package_manager\Exception\StageValidationException; use Drupal\package_manager\ValidationResult; use Drupal\package_manager_bypass\Committer; use Drupal\Tests\automatic_updates\Traits\EmailNotificationsTestTrait; +use Drupal\Tests\package_manager\Kernel\TestStage; use Drupal\Tests\package_manager\Traits\PackageManagerBypassTestTrait; use Drupal\Tests\user\Traits\UserCreationTrait; use Prophecy\Argument; @@ -344,6 +345,90 @@ class CronUpdaterTest extends AutomaticUpdatesKernelTestBase { } } + /** + * Tests stage is destroyed if not available and site is on insecure version. + */ + public function testStageDestroyedIfNotAvailable(): void { + $stage = $this->createStage(); + $stage->create(); + $original_stage_directory = $stage->getStageDirectory(); + $this->assertDirectoryExists($original_stage_directory); + + $listener = function (PostRequireEvent $event) use (&$cron_stage_dir, $original_stage_directory): void { + $this->assertDirectoryDoesNotExist($original_stage_directory); + $cron_stage_dir = $this->container->get('package_manager.stager')->getInvocationArguments()[0][1]->resolve(); + $this->assertSame($event->getStage()->getStageDirectory(), $cron_stage_dir); + $this->assertDirectoryExists($cron_stage_dir); + }; + $this->container->get('event_dispatcher')->addListener(PostRequireEvent::class, $listener, PHP_INT_MAX); + + $this->container->get('cron')->run(); + $this->assertIsString($cron_stage_dir); + $this->assertNotEquals($original_stage_directory, $cron_stage_dir); + $this->assertDirectoryDoesNotExist($cron_stage_dir); + $this->assertTrue($this->logger->hasRecord('The existing stage was not in the process of being applied, so it was destroyed to allow updating the site to a secure version during cron.', (string) RfcLogLevel::NOTICE)); + } + + /** + * Tests stage is not destroyed if another update is applying. + */ + public function testStageNotDestroyedIfApplying(): void { + $this->config('automatic_updates.settings')->set('cron', CronUpdater::ALL)->save(); + $this->setReleaseMetadata([ + 'drupal' => __DIR__ . "/../../../package_manager/tests/fixtures/release-history/drupal.9.8.1-security.xml", + ]); + $this->setCoreVersion('9.8.0'); + $stage = $this->createStage(); + $stage->create(); + $stage->require(['drupal/core:9.8.1']); + $stop_error = t('Stopping stage from applying'); + + // Add a PreApplyEvent event listener so we can attempt to run cron when + // another stage is applying. + $this->container->get('event_dispatcher')->addListener(PreApplyEvent::class, function (PreApplyEvent $event) use ($stop_error) { + // Ensure the stage that is applying the operation is not the cron + // updater. + $this->assertInstanceOf(TestStage::class, $event->getStage()); + $this->container->get('cron')->run(); + // We do not actually want to apply this operation it was just invoked to + // allow cron to be attempted. + $event->addError([$stop_error]); + }, PHP_INT_MAX); + + try { + $stage->apply(); + $this->fail('Expected update to fail'); + } + catch (StageValidationException $exception) { + $this->assertValidationResultsEqual([ValidationResult::createError([$stop_error])], $exception->getResults()); + } + + $this->assertTrue($this->logger->hasRecord("Cron will not perform any updates as an existing staged update is applying. The site is currently on an insecure version of Drupal core but will attempt to update to a secure version next time cron is run. This update may be applied manually at the <a href=\"%url\">update form</a>.", (string) RfcLogLevel::NOTICE)); + $this->assertUpdateStagedTimes(1); + } + + /** + * Tests stage is not destroyed if not available and site is on secure version. + */ + public function testStageNotDestroyedIfSecure(): void { + $this->config('automatic_updates.settings')->set('cron', CronUpdater::ALL)->save(); + $this->setReleaseMetadata([ + 'drupal' => __DIR__ . "/../../../package_manager/tests/fixtures/release-history/drupal.9.8.2.xml", + ]); + $this->setCoreVersion('9.8.1'); + $stage = $this->createStage(); + $stage->create(); + $stage->require(['drupal/random']); + $this->assertUpdateStagedTimes(1); + + // Trigger CronUpdater, the above should cause it to detect a stage that is + // applying. + $this->container->get('cron')->run(); + + $this->assertTrue($this->logger->hasRecord('Cron will not perform any updates because there is an existing stage and the current version of the site is secure.', (string) RfcLogLevel::NOTICE)); + $this->assertUpdateStagedTimes(1); + } + /** * Tests that CronUpdater::begin() unconditionally throws an exception. */