From 422fc62608b44249ba0069b743ab722a4258f933 Mon Sep 17 00:00:00 2001 From: "kunal.sachdev" <kunal.sachdev@3685163.no-reply.drupal.org> Date: Tue, 22 Mar 2022 12:35:16 +0000 Subject: [PATCH] Issue #3267577 by kunal.sachdev: Clean up in hook_uninstall --- automatic_updates.install | 7 ++ package_manager/package_manager.services.yml | 8 ++ .../src/PackageManagerUninstallValidator.php | 42 +++++++++ .../PackageManagerUninstallValidator.php | 88 +++++++++++++++++++ package_manager/src/Stage.php | 24 +++-- .../tests/src/Kernel/StageTest.php | 29 ++++++ 6 files changed, 192 insertions(+), 6 deletions(-) create mode 100644 package_manager/src/PackageManagerUninstallValidator.php create mode 100644 package_manager/src/ProxyClass/PackageManagerUninstallValidator.php diff --git a/automatic_updates.install b/automatic_updates.install index c17fad0c4c..02dbe7d529 100644 --- a/automatic_updates.install +++ b/automatic_updates.install @@ -7,6 +7,13 @@ use Drupal\automatic_updates\Validation\ReadinessRequirements; +/** + * Implements hook_uninstall(). + */ +function automatic_updates_uninstall() { + \Drupal::service('automatic_updates.updater')->destroy(TRUE); +} + /** * Implements hook_requirements(). */ diff --git a/package_manager/package_manager.services.yml b/package_manager/package_manager.services.yml index 271b0428c3..75ae35d2ec 100644 --- a/package_manager/package_manager.services.yml +++ b/package_manager/package_manager.services.yml @@ -137,3 +137,11 @@ services: - '@package_manager.path_locator' tags: - { name: event_subscriber } + package_manager.uninstall_validator: + class: Drupal\package_manager\PackageManagerUninstallValidator + tags: + - { name: module_install.uninstall_validator } + parent: container.trait + calls: + - ['setContainer', ['@service_container']] + lazy: true diff --git a/package_manager/src/PackageManagerUninstallValidator.php b/package_manager/src/PackageManagerUninstallValidator.php new file mode 100644 index 0000000000..dd9ab87607 --- /dev/null +++ b/package_manager/src/PackageManagerUninstallValidator.php @@ -0,0 +1,42 @@ +<?php + +namespace Drupal\package_manager; + +use Drupal\Core\Extension\ModuleUninstallValidatorInterface; +use Drupal\Core\StringTranslation\StringTranslationTrait; +use Symfony\Component\DependencyInjection\ContainerAwareInterface; +use Symfony\Component\DependencyInjection\ContainerAwareTrait; + +/** + * Prevents any module from being uninstalled if update is in process. + */ +class PackageManagerUninstallValidator implements ModuleUninstallValidatorInterface, ContainerAwareInterface { + + use ContainerAwareTrait; + use StringTranslationTrait; + + /** + * {@inheritdoc} + */ + public function validate($module) { + $stage = new Stage( + $this->container->get('config.factory'), + $this->container->get('package_manager.path_locator'), + $this->container->get('package_manager.beginner'), + $this->container->get('package_manager.stager'), + $this->container->get('package_manager.committer'), + $this->container->get('file_system'), + $this->container->get('event_dispatcher'), + $this->container->get('tempstore.shared'), + $this->container->get('datetime.time') + ); + if ($stage->isAvailable() || !$stage->isApplying()) { + return []; + } + if ($stage->isApplying()) { + $reasons[] = $this->t('Modules cannot be uninstalled while Package Manager is applying staged changes to the active code base.'); + } + return $reasons; + } + +} diff --git a/package_manager/src/ProxyClass/PackageManagerUninstallValidator.php b/package_manager/src/ProxyClass/PackageManagerUninstallValidator.php new file mode 100644 index 0000000000..275e878aa5 --- /dev/null +++ b/package_manager/src/ProxyClass/PackageManagerUninstallValidator.php @@ -0,0 +1,88 @@ +<?php +// phpcs:ignoreFile + +/** + * This file was generated via php core/scripts/generate-proxy-class.php 'Drupal\package_manager\PackageManagerUninstallValidator' "modules/contrib/automatic_updates/package_manager/src". + */ + +namespace Drupal\package_manager\ProxyClass { + + /** + * Provides a proxy class for \Drupal\package_manager\PackageManagerUninstallValidator. + * + * @see \Drupal\Component\ProxyBuilder + */ + class PackageManagerUninstallValidator implements \Drupal\Core\Extension\ModuleUninstallValidatorInterface + { + + use \Drupal\Core\DependencyInjection\DependencySerializationTrait; + + /** + * The id of the original proxied service. + * + * @var string + */ + protected $drupalProxyOriginalServiceId; + + /** + * The real proxied service, after it was lazy loaded. + * + * @var \Drupal\package_manager\PackageManagerUninstallValidator + */ + protected $service; + + /** + * The service container. + * + * @var \Symfony\Component\DependencyInjection\ContainerInterface + */ + protected $container; + + /** + * Constructs a ProxyClass Drupal proxy object. + * + * @param \Symfony\Component\DependencyInjection\ContainerInterface $container + * The container. + * @param string $drupal_proxy_original_service_id + * The service ID of the original service. + */ + public function __construct(\Symfony\Component\DependencyInjection\ContainerInterface $container, $drupal_proxy_original_service_id) + { + $this->container = $container; + $this->drupalProxyOriginalServiceId = $drupal_proxy_original_service_id; + } + + /** + * Lazy loads the real service from the container. + * + * @return object + * Returns the constructed real service. + */ + protected function lazyLoadItself() + { + if (!isset($this->service)) { + $this->service = $this->container->get($this->drupalProxyOriginalServiceId); + } + + return $this->service; + } + + /** + * {@inheritdoc} + */ + public function validate($module) + { + return $this->lazyLoadItself()->validate($module); + } + + /** + * {@inheritdoc} + */ + public function setStringTranslation(\Drupal\Core\StringTranslation\TranslationInterface $translation) + { + return $this->lazyLoadItself()->setStringTranslation($translation); + } + + } + +} diff --git a/package_manager/src/Stage.php b/package_manager/src/Stage.php index 71c4ceac26..810806fe74 100644 --- a/package_manager/src/Stage.php +++ b/package_manager/src/Stage.php @@ -359,12 +359,7 @@ class Stage { if (!$force) { $this->checkOwnership(); } - - // If we started applying staged changes to the active directory less than - // an hour ago, prevent the stage from being destroyed. - // @see :apply() - $apply_time = $this->tempStore->get(self::TEMPSTORE_APPLY_TIME_KEY); - if (isset($apply_time) && $this->time->getRequestTime() - $apply_time < 3600) { + if ($this->isApplying()) { throw new StageException('Cannot destroy the staging area while it is being applied to the active directory.'); } @@ -547,4 +542,21 @@ class Stage { return FileSystem::getOsTemporaryDirectory() . DIRECTORY_SEPARATOR . '.package_manager' . $site_id; } + /** + * Checks if staged changes are being applied to the active directory. + * + * @return bool + * TRUE if the staged changes are being applied to the active directory, and + * it has been less than an hour since that operation began. If more than an + * hour has elapsed since the changes started to be applied, FALSE is + * returned even if the stage internally thinks that changes are still being + * applied. + * + * @see ::apply() + */ + final public function isApplying(): bool { + $apply_time = $this->tempStore->get(self::TEMPSTORE_APPLY_TIME_KEY); + return isset($apply_time) && $this->time->getRequestTime() - $apply_time < 3600; + } + } diff --git a/package_manager/tests/src/Kernel/StageTest.php b/package_manager/tests/src/Kernel/StageTest.php index 90ac26e4d9..bf68cc8f39 100644 --- a/package_manager/tests/src/Kernel/StageTest.php +++ b/package_manager/tests/src/Kernel/StageTest.php @@ -4,6 +4,7 @@ namespace Drupal\Tests\package_manager\Kernel; use Drupal\Component\Datetime\Time; use Drupal\Core\DependencyInjection\ContainerBuilder; +use Drupal\Core\Extension\ModuleUninstallValidatorException; use Drupal\package_manager\Event\PostApplyEvent; use Drupal\package_manager\Event\PreApplyEvent; use Drupal\package_manager\Event\StageEvent; @@ -12,6 +13,8 @@ use Drupal\package_manager\Exception\StageException; /** * @coversDefaultClass \Drupal\package_manager\Stage * + * @covers \Drupal\package_manager\PackageManagerUninstallValidator + * * @group package_manager */ class StageTest extends PackageManagerKernelTestBase { @@ -179,6 +182,32 @@ class StageTest extends PackageManagerKernelTestBase { $stage->apply(); } + /** + * Test uninstalling any module while the staged changes are being applied. + */ + public function testUninstallModuleDuringApply(): void { + $listener = function (PreApplyEvent $event): void { + $this->assertTrue($event->getStage()->isApplying()); + + // Trying to uninstall any module while the stage is being applied should + // result in a module uninstall validation error. + try { + $this->container->get('module_installer') + ->uninstall(['package_manager_bypass']); + $this->fail('Expected an exception to be thrown while uninstalling a module.'); + } + catch (ModuleUninstallValidatorException $e) { + $this->assertStringContainsString('Modules cannot be uninstalled while Package Manager is applying staged changes to the active code base.', $e->getMessage()); + } + }; + $this->container->get('event_dispatcher') + ->addListener(PreApplyEvent::class, $listener); + + $stage = $this->createStage(); + $stage->create(); + $stage->apply(); + } + } /** -- GitLab