diff --git a/src/BatchProcessor.php b/src/BatchProcessor.php index 8723b9ce9ccbb818889d2ce6b9f7aa0dc856eb22..09a5ac6b94d9fed301a51c49b0f752a7ff7e89e7 100644 --- a/src/BatchProcessor.php +++ b/src/BatchProcessor.php @@ -57,14 +57,16 @@ class BatchProcessor { /** * Calls the updater's begin() method. * + * @param string[] $project_versions + * The project versions to be staged in the update, keyed by package name. * @param array $context * The current context of the batch job. * * @see \Drupal\automatic_updates\Updater::begin() */ - public static function begin(array &$context): void { + public static function begin(array $project_versions, array &$context): void { try { - static::getUpdater()->begin(); + static::getUpdater()->begin($project_versions); } catch (\Throwable $e) { static::handleException($e, $context); @@ -74,16 +76,14 @@ class BatchProcessor { /** * Calls the updater's stageVersions() method. * - * @param string[] $project_versions - * The project versions to be staged in the update, keyed by package name. * @param array $context * The current context of the batch job. * - * @see \Drupal\automatic_updates\Updater::stageVersions() + * @see \Drupal\automatic_updates\Updater::stage() */ - public static function stageProjectVersions(array $project_versions, array &$context): void { + public static function stage(array &$context): void { try { - static::getUpdater()->stageVersions($project_versions); + static::getUpdater()->stage(); } catch (\Throwable $e) { static::handleException($e, $context); diff --git a/src/Form/UpdaterForm.php b/src/Form/UpdaterForm.php index 320f6ebc63da69fe5dd0cd31da95b7dc1f6e1038..36d4c8c6e02d369f8690aa417220aa6694e1bf43 100644 --- a/src/Form/UpdaterForm.php +++ b/src/Form/UpdaterForm.php @@ -235,10 +235,11 @@ class UpdaterForm extends FormBase { $batch = (new BatchBuilder()) ->setTitle($this->t('Downloading updates')) ->setInitMessage($this->t('Preparing to download updates')) - ->addOperation([BatchProcessor::class, 'begin']) - ->addOperation([BatchProcessor::class, 'stageProjectVersions'], [ - $form_state->getValue('update_version'), - ]) + ->addOperation( + [BatchProcessor::class, 'begin'], + [$form_state->getValue('update_version')] + ) + ->addOperation([BatchProcessor::class, 'stage']) ->setFinishCallback([BatchProcessor::class, 'finishStage']) ->toArray(); diff --git a/src/Updater.php b/src/Updater.php index 1da8d0cd41b31ae666df34ee84568a27d314e0d4..5c8fbce6783ddd367b28f5b6b38701702cf92de6 100644 --- a/src/Updater.php +++ b/src/Updater.php @@ -163,11 +163,21 @@ class Updater { /** * Begins the update. * + * @param string[] $project_versions + * The versions of the packages to update to, keyed by package name. + * * @return string * A key for this stage update process. + * + * @throws \InvalidArgumentException + * Thrown if no project version for Drupal core is provided. */ - public function begin(): string { - $stage_key = $this->createActiveStage(); + public function begin(array $project_versions): string { + if (count($project_versions) !== 1 || !array_key_exists('drupal', $project_versions)) { + throw new \InvalidArgumentException("Currently only updates to Drupal core are supported."); + } + $packages[] = 'drupal/core:' . $project_versions['drupal']; + $stage_key = $this->createActiveStage($packages); $event = $this->dispatchUpdateEvent(AutomaticUpdatesEvents::PRE_START); $this->beginner->begin(static::getActiveDirectory(), static::getStageDirectory(), $this->getExclusions($event)); return $stage_key; @@ -190,24 +200,11 @@ class Updater { } /** - * Adds specific project versions to the staging area. - * - * @param string[] $project_versions - * The project versions to add to the staging area, keyed by package name. + * Stages the update. */ - public function stageVersions(array $project_versions): void { - $packages = []; - foreach ($project_versions as $project => $project_version) { - if ($project === 'drupal') { - // @todo Determine when to use drupal/core-recommended and when to use - // drupal/core - $packages[] = "drupal/core:$project_version"; - } - else { - $packages[] = "drupal/$project:$project_version"; - } - } - $this->stagePackages($packages); + public function stage(): void { + $current = $this->state->get(static::STATE_KEY); + $this->stagePackages($current['package_versions']); } /** @@ -220,11 +217,6 @@ class Updater { $command = array_merge(['require'], $packages); $command[] = '--update-with-all-dependencies'; $this->stageCommand($command); - // Store the expected packages to confirm no other Drupal packages were - // updated. - $current = $this->state->get(static::STATE_KEY); - $current['packages'] = $packages; - $this->state->set(self::STATE_KEY, $current); } /** @@ -265,12 +257,21 @@ class Updater { /** * Initializes an active update and returns its ID. * + * @param string[] $package_versions + * The versions of the packages to stage, keyed by package name. + * * @return string * The active update ID. */ - private function createActiveStage(): string { + private function createActiveStage(array $package_versions): string { $value = static::STATE_KEY . microtime(); - $this->state->set(static::STATE_KEY, ['id' => $value]); + $this->state->set( + static::STATE_KEY, + [ + 'id' => $value, + 'package_versions' => $package_versions, + ] + ); return $value; } diff --git a/tests/src/Functional/ExclusionsTest.php b/tests/src/Functional/ExclusionsTest.php index 0b978a5c34895e785f35a20fd9f9aecef3302bef..26d8a26cdaad47b236c9bc4ceaacf8921ea8e686 100644 --- a/tests/src/Functional/ExclusionsTest.php +++ b/tests/src/Functional/ExclusionsTest.php @@ -40,7 +40,7 @@ class ExclusionsTest extends BrowserTestBase { $settings['file_private_path'] = 'files/private'; new Settings($settings); - $updater->begin(); + $updater->begin(['drupal' => '9.8.1']); $this->assertFileDoesNotExist("$stage_dir/sites/default/settings.php"); $this->assertFileDoesNotExist("$stage_dir/sites/default/settings.local.php"); $this->assertFileDoesNotExist("$stage_dir/sites/default/services.yml"); diff --git a/tests/src/Kernel/UpdaterTest.php b/tests/src/Kernel/UpdaterTest.php new file mode 100644 index 0000000000000000000000000000000000000000..8f8ab0ecb28e461e8cdda72a7feefd6c3220a207 --- /dev/null +++ b/tests/src/Kernel/UpdaterTest.php @@ -0,0 +1,72 @@ +<?php + +namespace Drupal\Tests\automatic_updates\Kernel; + +use Drupal\KernelTests\KernelTestBase; +use Prophecy\Argument; + +/** + * @coversDefaultClass \Drupal\automatic_updates\Updater + * + * @group automatic_updates + */ +class UpdaterTest extends KernelTestBase { + + /** + * {@inheritdoc} + */ + protected static $modules = [ + 'automatic_updates', + 'update', + 'composer_stager_bypass', + ]; + + /** + * Tests that correct versions are staged after calling ::begin(). + */ + public function testCorrectVersionsStaged() { + $this->container->get('automatic_updates.updater')->begin([ + 'drupal' => '9.8.1', + ]); + // Rebuild the container to ensure the project versions are kept in state. + /** @var \Drupal\Core\DrupalKernel $kernel */ + $kernel = $this->container->get('kernel'); + $kernel->rebuildContainer(); + $this->container = $kernel->getContainer(); + $stager = $this->prophesize('\PhpTuf\ComposerStager\Domain\StagerInterface'); + $command = [ + 'require', + 'drupal/core:9.8.1', + '--update-with-all-dependencies', + ]; + $stager->stage($command, Argument::cetera())->shouldBeCalled(); + $this->container->set('automatic_updates.stager', $stager->reveal()); + $this->container->get('automatic_updates.updater')->stage(); + } + + /** + * @covers ::begin + * + * @dataProvider providerInvalidProjectVersions + */ + public function testInvalidProjectVersions(array $project_versions): void { + $this->expectException(\InvalidArgumentException::class); + $this->expectExceptionMessage('Currently only updates to Drupal core are supported.'); + $this->container->get('automatic_updates.updater')->begin($project_versions); + } + + /** + * Data provider for testInvalidProjectVersions(). + * + * @return array + * The test cases for testInvalidProjectVersions(). + */ + public function providerInvalidProjectVersions(): array { + return [ + 'only not drupal' => [['not_drupal' => '1.1.3']], + 'not drupal and drupal' => [['drupal' => '9.8.0', 'not_drupal' => '1.2.3']], + 'empty' => [[]], + ]; + } + +}