From ab695b5cc6d8a02eea7d593573cebf388e2e99f4 Mon Sep 17 00:00:00 2001 From: Alex Pott <alex.a.pott@googlemail.com> Date: Fri, 8 Mar 2024 14:03:20 +0000 Subject: [PATCH] Issue #3426324 by Wim Leers: ExistsConstraintValidator should ignore NULL values and treat `core` as a valid module --- .../ExtensionExistsConstraintValidator.php | 14 ++++++++++++ ...ExtensionExistsConstraintValidatorTest.php | 22 ++++++++++++++++++- 2 files changed, 35 insertions(+), 1 deletion(-) diff --git a/core/lib/Drupal/Core/Extension/Plugin/Validation/Constraint/ExtensionExistsConstraintValidator.php b/core/lib/Drupal/Core/Extension/Plugin/Validation/Constraint/ExtensionExistsConstraintValidator.php index 404d5dcf38c2..6f4599618901 100644 --- a/core/lib/Drupal/Core/Extension/Plugin/Validation/Constraint/ExtensionExistsConstraintValidator.php +++ b/core/lib/Drupal/Core/Extension/Plugin/Validation/Constraint/ExtensionExistsConstraintValidator.php @@ -61,12 +61,26 @@ public function validate(mixed $extension_name, Constraint $constraint) { switch ($constraint->type) { case 'module': + // This constraint may be used to validate nullable (optional) values. + if ($extension_name === NULL) { + return; + } + // Some plugins are shipped in `core/lib`, which corresponds to the + // special `core` extension name. + // For example: \Drupal\Core\Menu\Plugin\Block\LocalActionsBlock. + if ($extension_name === 'core') { + return; + } if (!$this->moduleHandler->moduleExists($extension_name)) { $this->context->addViolation($constraint->moduleMessage, $variables); } break; case 'theme': + // This constraint may be used to validate nullable (optional) values. + if ($extension_name === NULL) { + return; + } if (!$this->themeHandler->themeExists($extension_name)) { $this->context->addViolation($constraint->themeMessage, $variables); } diff --git a/core/tests/Drupal/KernelTests/Core/Extension/ExtensionExistsConstraintValidatorTest.php b/core/tests/Drupal/KernelTests/Core/Extension/ExtensionExistsConstraintValidatorTest.php index f698279a56d6..9fec10ddff08 100644 --- a/core/tests/Drupal/KernelTests/Core/Extension/ExtensionExistsConstraintValidatorTest.php +++ b/core/tests/Drupal/KernelTests/Core/Extension/ExtensionExistsConstraintValidatorTest.php @@ -31,8 +31,13 @@ public function testValidation(): void { /** @var \Drupal\Core\TypedData\TypedDataManagerInterface $typed_data */ $typed_data = $this->container->get('typed_data_manager'); - $data = $typed_data->create($definition, 'user'); + // `core` provides many plugins without the need to install a module. + $data = $typed_data->create($definition, 'core'); + $violations = $data->validate(); + $this->assertCount(0, $violations); + + $data->setValue('user'); $violations = $data->validate(); $this->assertCount(1, $violations); $this->assertSame("Module 'user' is not installed.", (string) $violations->get(0)->getMessage()); @@ -40,6 +45,10 @@ public function testValidation(): void { $this->enableModules(['user']); $this->assertCount(0, $data->validate()); + // NULL should not trigger a validation error: a value may be nullable. + $data->setValue(NULL); + $this->assertCount(0, $data->validate()); + $definition->setConstraints(['ExtensionExists' => 'theme']); $data = $typed_data->create($definition, 'stark'); @@ -56,6 +65,17 @@ public function testValidation(): void { ->create($definition, 'stark'); $this->assertCount(0, $data->validate()); + // `core` provides many plugins without the need to install a module, but it + // does not work for themes. + $data = $typed_data->create($definition, 'core'); + $violations = $data->validate(); + $this->assertCount(1, $violations); + $this->assertSame("Theme 'core' is not installed.", (string) $violations->get(0)->getMessage()); + + // NULL should not trigger a validation error: a value may be nullable. + $data->setValue(NULL); + $this->assertCount(0, $data->validate()); + // Anything but a module or theme should raise an exception. $definition->setConstraints(['ExtensionExists' => 'profile']); $this->expectExceptionMessage("Unknown extension type: 'profile'"); -- GitLab