
| Current Path : /var/www/html/stolberg/web/core/tests/Drupal/KernelTests/Core/Theme/ |
Linux ift1.ift-informatik.de 5.4.0-216-generic #236-Ubuntu SMP Fri Apr 11 19:53:21 UTC 2025 x86_64 |
| Current File : /var/www/html/stolberg/web/core/tests/Drupal/KernelTests/Core/Theme/ThemeInstallerTest.php |
<?php
declare(strict_types=1);
namespace Drupal\KernelTests\Core\Theme;
use Drupal\Core\DependencyInjection\ContainerBuilder;
use Drupal\Core\Extension\ExtensionNameLengthException;
use Drupal\Core\Extension\ExtensionNameReservedException;
use Drupal\Core\Extension\MissingDependencyException;
use Drupal\Core\Extension\ModuleUninstallValidatorException;
use Drupal\Core\Extension\Exception\UnknownExtensionException;
use Drupal\Core\Extension\ThemeExtensionList;
use Drupal\KernelTests\KernelTestBase;
/**
* Tests installing and uninstalling of themes.
*
* @group Extension
*/
class ThemeInstallerTest extends KernelTestBase {
/**
* {@inheritdoc}
*/
protected static $modules = ['system'];
/**
* {@inheritdoc}
*/
public function register(ContainerBuilder $container) {
parent::register($container);
// Some test methods involve ModuleHandler operations, which attempt to
// rebuild and dump routes.
$container
->register('router.dumper', 'Drupal\Core\Routing\NullMatcherDumper');
}
/**
* {@inheritdoc}
*/
protected function setUp(): void {
parent::setUp();
$this->installConfig(['system']);
}
/**
* Verifies that no themes are installed by default.
*/
public function testEmpty(): void {
$this->assertEmpty($this->extensionConfig()->get('theme'));
$this->assertEmpty(array_keys($this->themeHandler()->listInfo()));
$this->assertEmpty(array_keys(\Drupal::service('theme_handler')->listInfo()));
// Rebuilding available themes should always yield results though.
$this->assertNotEmpty($this->extensionListTheme()->reset()->getList()['stark'], 'ThemeExtensionList::getList() yields all available themes.');
// theme_get_setting() should return global default theme settings.
$this->assertTrue(theme_get_setting('features.favicon'));
}
/**
* Tests installing a theme.
*/
public function testInstall(): void {
$name = 'test_basetheme';
$themes = $this->themeHandler()->listInfo();
$this->assertFalse(isset($themes[$name]));
$this->themeInstaller()->install([$name]);
$this->assertSame(0, $this->extensionConfig()->get("theme.{$name}"));
$themes = $this->themeHandler()->listInfo();
$this->assertTrue(isset($themes[$name]));
$this->assertEquals($name, $themes[$name]->getName());
// Verify that test_basetheme.settings is active.
$this->assertFalse(theme_get_setting('features.favicon', $name));
$this->assertEquals('only', theme_get_setting('base', $name));
$this->assertEquals('base', theme_get_setting('override', $name));
}
/**
* Tests installing a sub-theme.
*/
public function testInstallSubTheme(): void {
$name = 'test_child_theme';
$base_name = 'test_parent_theme';
$themes = $this->themeHandler()->listInfo();
$this->assertEmpty(array_keys($themes));
$this->themeInstaller()->install([$name]);
$themes = $this->themeHandler()->listInfo();
$this->assertTrue(isset($themes[$name]));
$this->assertTrue(isset($themes[$base_name]));
$expectedOrder = [
$base_name,
$name,
];
$this->assertEquals($expectedOrder, array_keys($themes));
$this->themeInstaller()->uninstall([$name]);
$themes = $this->themeHandler()->listInfo();
$this->assertFalse(isset($themes[$name]));
$this->assertTrue(isset($themes[$base_name]));
}
/**
* Tests installing a non-existing theme.
*/
public function testInstallNonExisting(): void {
$name = 'non_existing_theme';
$themes = $this->themeHandler()->listInfo();
$this->assertEmpty(array_keys($themes));
try {
$message = 'ThemeInstaller::install() throws UnknownExtensionException upon installing a non-existing theme.';
$this->themeInstaller()->install([$name]);
$this->fail($message);
}
catch (\Exception $e) {
$this->assertInstanceOf(UnknownExtensionException::class, $e);
}
$themes = $this->themeHandler()->listInfo();
$this->assertEmpty(array_keys($themes));
}
/**
* Tests installing a theme with a too long name.
*/
public function testInstallNameTooLong(): void {
$name = 'test_theme_having_veery_long_name_which_is_too_long';
try {
$message = 'ThemeInstaller::install() throws ExtensionNameLengthException upon installing a theme with a too long name.';
$this->themeInstaller()->install([$name]);
$this->fail($message);
}
catch (\Exception $e) {
$this->assertInstanceOf(ExtensionNameLengthException::class, $e);
}
}
/**
* Tests installing a theme with the same name as an enabled module.
*/
public function testInstallThemeSameNameAsModule(): void {
$name = 'name_collision_test';
// Install and uninstall the theme.
$this->themeInstaller()->install([$name]);
$this->themeInstaller()->uninstall([$name]);
// Install the module, then the theme.
$this->moduleInstaller()->install([$name]);
$message = "Theme name {$name} is already in use by an installed module.";
$this->expectException(ExtensionNameReservedException::class);
$this->expectExceptionMessage($message);
$this->themeInstaller()->install([$name]);
}
/**
* Tests installing a theme with unmet module dependencies.
*
* @dataProvider providerTestInstallThemeWithUnmetModuleDependencies
*/
public function testInstallThemeWithUnmetModuleDependencies($theme_name, $installed_modules, $message): void {
$this->moduleInstaller()->install($installed_modules);
$themes = $this->themeHandler()->listInfo();
$this->assertEmpty($themes);
$this->expectException(MissingDependencyException::class);
$this->expectExceptionMessage($message);
$this->themeInstaller()->install([$theme_name]);
}
/**
* Tests trying to install a deprecated theme.
*
* @covers \Drupal\Core\Extension\ThemeInstaller::install
*
* @group legacy
*/
public function testInstallDeprecated(): void {
$this->expectDeprecation("The theme 'deprecated_theme_test' is deprecated. See https://example.com/deprecated");
$this->themeInstaller()->install(['deprecated_theme_test']);
$this->assertTrue(\Drupal::service('theme_handler')->themeExists('deprecated_theme_test'));
}
/**
* Data provider for testInstallThemeWithUnmetModuleDependencies().
*/
public static function providerTestInstallThemeWithUnmetModuleDependencies() {
return [
'theme with uninstalled module dependencies' => [
'test_theme_depending_on_modules',
[],
"Unable to install theme: 'test_theme_depending_on_modules' due to unmet module dependencies: 'test_module_required_by_theme, test_another_module_required_by_theme'.",
],
'theme with a base theme with uninstalled module dependencies' => [
'test_theme_with_a_base_theme_depending_on_modules',
[],
"Unable to install theme: 'test_theme_with_a_base_theme_depending_on_modules' due to unmet module dependencies: 'test_module_required_by_theme, test_another_module_required_by_theme'.",
],
'theme and base theme have uninstalled module dependencies' => [
'test_theme_mixed_module_dependencies',
[],
"Unable to install theme: 'test_theme_mixed_module_dependencies' due to unmet module dependencies: 'help, test_module_required_by_theme, test_another_module_required_by_theme'.",
],
'theme with already installed module dependencies, base theme module dependencies are not installed' => [
'test_theme_mixed_module_dependencies',
['help'],
"Unable to install theme: 'test_theme_mixed_module_dependencies' due to unmet module dependencies: 'test_module_required_by_theme, test_another_module_required_by_theme'.",
],
'theme with module dependencies not installed, base theme module dependencies are already installed, ' => [
'test_theme_mixed_module_dependencies',
['test_module_required_by_theme', 'test_another_module_required_by_theme'],
"Unable to install theme: 'test_theme_mixed_module_dependencies' due to unmet module dependencies: 'help'.",
],
'theme depending on a module that does not exist' => [
'test_theme_depending_on_nonexisting_module',
[],
"Unable to install theme: 'test_theme_depending_on_nonexisting_module' due to unmet module dependencies: 'test_module_non_existing",
],
'theme depending on an installed but incompatible module' => [
'test_theme_depending_on_constrained_modules',
['test_module_compatible_constraint', 'test_module_incompatible_constraint'],
"Unable to install theme: Test Module Theme Depends on with Incompatible Constraint (>=8.x-2.x) (incompatible with version 8.x-1.8)",
],
];
}
/**
* Tests installing a theme with module dependencies that are met.
*/
public function testInstallThemeWithMetModuleDependencies(): void {
$name = 'test_theme_depending_on_modules';
$themes = $this->themeHandler()->listInfo();
$this->assertArrayNotHasKey($name, $themes);
$this->moduleInstaller()->install(['test_module_required_by_theme', 'test_another_module_required_by_theme']);
$this->themeInstaller()->install([$name]);
$themes = $this->themeHandler()->listInfo();
$this->assertArrayHasKey($name, $themes);
$this->expectException(ModuleUninstallValidatorException::class);
$this->expectExceptionMessage('The following reasons prevent the modules from being uninstalled: Required by the theme: Test Theme Depending on Modules');
$this->moduleInstaller()->uninstall(['test_module_required_by_theme']);
}
/**
* Tests uninstalling the default theme.
*/
public function testUninstallDefault(): void {
$name = 'stark';
$other_name = 'olivero';
$this->themeInstaller()->install([$name, $other_name]);
$this->config('system.theme')->set('default', $name)->save();
$themes = $this->themeHandler()->listInfo();
$this->assertTrue(isset($themes[$name]));
$this->assertTrue(isset($themes[$other_name]));
try {
$message = 'ThemeInstaller::uninstall() throws InvalidArgumentException upon uninstalling default theme.';
$this->themeInstaller()->uninstall([$name]);
$this->fail($message);
}
catch (\Exception $e) {
$this->assertInstanceOf(\InvalidArgumentException::class, $e);
}
$themes = $this->themeHandler()->listInfo();
$this->assertTrue(isset($themes[$name]));
$this->assertTrue(isset($themes[$other_name]));
}
/**
* Tests uninstalling the admin theme.
*/
public function testUninstallAdmin(): void {
$name = 'stark';
$other_name = 'olivero';
$this->themeInstaller()->install([$name, $other_name]);
$this->config('system.theme')->set('admin', $name)->save();
$themes = $this->themeHandler()->listInfo();
$this->assertTrue(isset($themes[$name]));
$this->assertTrue(isset($themes[$other_name]));
try {
$message = 'ThemeInstaller::uninstall() throws InvalidArgumentException upon disabling admin theme.';
$this->themeInstaller()->uninstall([$name]);
$this->fail($message);
}
catch (\Exception $e) {
$this->assertInstanceOf(\InvalidArgumentException::class, $e);
}
$themes = $this->themeHandler()->listInfo();
$this->assertTrue(isset($themes[$name]));
$this->assertTrue(isset($themes[$other_name]));
}
/**
* Tests uninstalling a sub-theme.
*/
public function testUninstallSubTheme(): void {
$name = 'test_subtheme';
$base_name = 'test_basetheme';
$this->themeInstaller()->install([$name]);
$this->themeInstaller()->uninstall([$name]);
$themes = $this->themeHandler()->listInfo();
$this->assertFalse(isset($themes[$name]));
$this->assertTrue(isset($themes[$base_name]));
}
/**
* Tests uninstalling a base theme before its sub-theme.
*/
public function testUninstallBaseBeforeSubTheme(): void {
$name = 'test_basetheme';
$sub_name = 'test_subtheme';
$this->themeInstaller()->install([$sub_name]);
try {
$message = 'ThemeInstaller::install() throws InvalidArgumentException upon uninstalling base theme before sub theme.';
$this->themeInstaller()->uninstall([$name]);
$this->fail($message);
}
catch (\Exception $e) {
$this->assertInstanceOf(\InvalidArgumentException::class, $e);
}
$themes = $this->themeHandler()->listInfo();
$this->assertTrue(isset($themes[$name]));
$this->assertTrue(isset($themes[$sub_name]));
// Verify that uninstalling both at the same time works.
$this->themeInstaller()->uninstall([$name, $sub_name]);
$themes = $this->themeHandler()->listInfo();
$this->assertFalse(isset($themes[$name]));
$this->assertFalse(isset($themes[$sub_name]));
}
/**
* Tests uninstalling a non-existing theme.
*/
public function testUninstallNonExisting(): void {
$name = 'non_existing_theme';
$themes = $this->themeHandler()->listInfo();
$this->assertEmpty(array_keys($themes));
$this->expectException(UnknownExtensionException::class);
$this->themeInstaller()->uninstall([$name]);
}
/**
* Tests uninstalling a theme.
*/
public function testUninstall(): void {
$name = 'test_basetheme';
$this->themeInstaller()->install([$name]);
$this->assertNotEmpty($this->config("$name.settings")->get());
$this->themeInstaller()->uninstall([$name]);
$this->assertEmpty(array_keys($this->themeHandler()->listInfo()));
$this->assertEmpty($this->config("$name.settings")->get());
// Ensure that the uninstalled theme can be installed again.
$this->themeInstaller()->install([$name]);
$themes = $this->themeHandler()->listInfo();
$this->assertTrue(isset($themes[$name]));
$this->assertEquals($name, $themes[$name]->getName());
$this->assertNotEmpty($this->config("$name.settings")->get());
}
/**
* Tests uninstalling a theme that is not installed.
*/
public function testUninstallNotInstalled(): void {
$name = 'test_basetheme';
$themes = $this->themeHandler()->listInfo();
$this->assertEmpty(array_keys($themes));
$this->expectException(UnknownExtensionException::class);
$this->themeInstaller()->uninstall([$name]);
}
/**
* Tests that theme info can be altered by a module.
*
* @see module_test_system_info_alter()
*/
public function testThemeInfoAlter(): void {
$name = 'stark';
$this->container->get('state')->set('module_test.hook_system_info_alter', TRUE);
$this->themeInstaller()->install([$name]);
$themes = $this->themeHandler()->listInfo();
$this->assertFalse(isset($themes[$name]->info['regions']['test_region']));
// Install module_test.
$this->moduleInstaller()->install(['module_test'], FALSE);
$this->assertTrue($this->moduleHandler()->moduleExists('module_test'));
$themes = $this->themeHandler()->listInfo();
$this->assertTrue(isset($themes[$name]->info['regions']['test_region']));
// Legacy assertions.
// @todo Remove once theme initialization/info has been modernized.
// @see https://www.drupal.org/node/2228093
$info = \Drupal::service('extension.list.theme')->getExtensionInfo($name);
$this->assertTrue(isset($info['regions']['test_region']));
$regions = system_region_list($name);
$this->assertTrue(isset($regions['test_region']));
$theme_list = \Drupal::service('theme_handler')->listInfo();
$this->assertTrue(isset($theme_list[$name]->info['regions']['test_region']));
$this->moduleInstaller()->uninstall(['module_test']);
$this->assertFalse($this->moduleHandler()->moduleExists('module_test'));
$themes = $this->themeHandler()->listInfo();
$this->assertFalse(isset($themes[$name]->info['regions']['test_region']));
// Legacy assertions.
// @todo Remove once theme initialization/info has been modernized.
// @see https://www.drupal.org/node/2228093
$info = \Drupal::service('extension.list.theme')->getExtensionInfo($name);
$this->assertFalse(isset($info['regions']['test_region']));
$regions = system_region_list($name);
$this->assertFalse(isset($regions['test_region']));
$theme_list = \Drupal::service('theme_handler')->listInfo();
$this->assertFalse(isset($theme_list[$name]->info['regions']['test_region']));
}
/**
* Returns the theme handler service.
*
* @return \Drupal\Core\Extension\ThemeHandlerInterface
*/
protected function themeHandler() {
return $this->container->get('theme_handler');
}
/**
* Returns the theme installer service.
*
* @return \Drupal\Core\Extension\ThemeInstallerInterface
*/
protected function themeInstaller() {
return $this->container->get('theme_installer');
}
/**
* Returns the system.theme config object.
*
* @return \Drupal\Core\Config\Config
*/
protected function extensionConfig() {
return $this->config('core.extension');
}
/**
* Returns the ModuleHandler.
*
* @return \Drupal\Core\Extension\ModuleHandlerInterface
*/
protected function moduleHandler() {
return $this->container->get('module_handler');
}
/**
* Returns the ModuleInstaller.
*
* @return \Drupal\Core\Extension\ModuleInstallerInterface
*/
protected function moduleInstaller() {
return $this->container->get('module_installer');
}
/**
* Returns the ThemeExtensionList.
*
* @return \Drupal\Core\Extension\ThemeExtensionList
*/
protected function extensionListTheme(): ThemeExtensionList {
return $this->container->get('extension.list.theme');
}
}