Skip to content
Snippets Groups Projects
Verified Commit fa46ee8f authored by Lee Rowlands's avatar Lee Rowlands
Browse files

Issue #3405800 by alexpott, phenaproxima, Wim Leers, bircher, larowlan: Config...

Issue #3405800 by alexpott, phenaproxima, Wim Leers, bircher, larowlan: Config collections do not trigger configuration events consistently
parent 59abed4d
No related branches found
No related tags found
No related merge requests found
Showing
with 311 additions and 39 deletions
......@@ -226,7 +226,8 @@ public function save($has_trusted_data = FALSE) {
Cache::invalidateTags($this->getCacheTags());
}
$this->isNew = FALSE;
$this->eventDispatcher->dispatch(new ConfigCrudEvent($this), ConfigEvents::SAVE);
$event_name = $this->getStorage()->getCollectionName() === StorageInterface::DEFAULT_COLLECTION ? ConfigEvents::SAVE : ConfigCollectionEvents::SAVE_IN_COLLECTION;
$this->eventDispatcher->dispatch(new ConfigCrudEvent($this), $event_name);
$this->originalData = $this->data;
return $this;
}
......@@ -243,21 +244,12 @@ public function delete() {
Cache::invalidateTags($this->getCacheTags());
$this->isNew = TRUE;
$this->resetOverriddenData();
$this->eventDispatcher->dispatch(new ConfigCrudEvent($this), ConfigEvents::DELETE);
$event_name = $this->getStorage()->getCollectionName() === StorageInterface::DEFAULT_COLLECTION ? ConfigEvents::DELETE : ConfigCollectionEvents::DELETE_IN_COLLECTION;
$this->eventDispatcher->dispatch(new ConfigCrudEvent($this), $event_name);
$this->originalData = $this->data;
return $this;
}
/**
* Gets the raw data without overrides.
*
* @return array
* The raw data.
*/
public function getRawData() {
return $this->data;
}
/**
* Gets original data from this configuration object.
*
......
<?php
namespace Drupal\Core\Config;
/**
* Defines events for working with configuration collections.
*
* Configuration collections are often used to store configuration-related
* data, like overrides. The use case is determined by the module that provides
* the collection. A classic example is to store the translated parts of
* various configuration objects. Using a collection allows this data to be
* imported and exported alongside regular configuration. It also allows the
* data to be created when installing an extension. In both the import/export
* and extension installation situations, collection data is stored in
* subdirectories.
*
* @see \Drupal\Core\Config\ConfigCrudEvent
*/
final class ConfigCollectionEvents {
/**
* Event dispatched when saving configuration not in the default collection.
*
* This event allows modules to react whenever an object that extends
* \Drupal\Core\Config\StorableConfigBase is saved in a non-default
* collection. The event listener method receives a
* \Drupal\Core\Config\ConfigCrudEvent instance.
*
* Note: this event is not used for configuration in the default collection.
* See \Drupal\Core\Config\ConfigEvents::SAVE instead.
*
* @Event
*
* @var string
*
* @see \Drupal\Core\Config\ConfigCrudEvent
* @see \Drupal\Core\Config\ConfigFactoryOverrideInterface::createConfigObject()
* @see \Drupal\language\Config\LanguageConfigOverride::save()
*
* @see \Drupal\Core\Config\ConfigEvents::SAVE
*/
const SAVE_IN_COLLECTION = 'config.save.collection';
/**
* Event dispatched when deleting configuration not in the default collection.
*
* This event allows modules to react whenever an object that extends
* \Drupal\Core\Config\StorableConfigBase is deleted in a non-default
* collection. The event listener method receives a
* \Drupal\Core\Config\ConfigCrudEvent instance.
*
* Note: this event is not used for configuration in the default collection.
* See \Drupal\Core\Config\ConfigEvents::DELETE instead.
*
* @Event
*
* @see \Drupal\Core\Config\ConfigEvents::DELETE
* @see \Drupal\Core\Config\ConfigCrudEvent
* @see \Drupal\Core\Config\ConfigFactoryOverrideInterface::createConfigObject()
* @see \Drupal\language\Config\LanguageConfigOverride::delete()
*
* @var string
*/
const DELETE_IN_COLLECTION = 'config.delete.collection';
/**
* Event dispatched when renaming configuration not in the default collection.
*
* This event allows modules to react whenever an object that extends
* \Drupal\Core\Config\StorableConfigBase is renamed in a non-default
* collection. The event listener method receives a
* \Drupal\Core\Config\ConfigCrudEvent instance.
*
* Note: this event is not used for configuration in the default collection.
* See \Drupal\Core\Config\ConfigEvents::RENAME instead.
*
* @Event
*
* @see \Drupal\Core\Config\ConfigEvents::RENAME
* @see \Drupal\Core\Config\ConfigCrudEvent
* @see \Drupal\Core\Config\ConfigFactoryOverrideInterface::createConfigObject()
*
* @var string
*/
const RENAME_IN_COLLECTION = 'config.rename.collection';
/**
* Event dispatched to collect information on all config collections.
*
* This event allows modules to add to the list of configuration collections
* retrieved by \Drupal\Core\Config\ConfigManager::getConfigCollectionInfo().
* The event listener method receives a
* \Drupal\Core\Config\ConfigCollectionInfo instance.
*
* @Event
*
* @see \Drupal\Core\Config\ConfigCollectionInfo
* @see \Drupal\Core\Config\ConfigManager::getConfigCollectionInfo()
* @see \Drupal\Core\Config\ConfigFactoryOverrideBase
*
* @var string
*/
const COLLECTION_INFO = 'config.collection_info';
}
......@@ -19,17 +19,17 @@ class ConfigCrudEvent extends Event {
/**
* Constructs a configuration event object.
*
* @param \Drupal\Core\Config\Config $config
* @param \Drupal\Core\Config\StorableConfigBase $config
* Configuration object.
*/
public function __construct(Config $config) {
public function __construct(StorableConfigBase $config) {
$this->config = $config;
}
/**
* Gets configuration object.
*
* @return \Drupal\Core\Config\Config
* @return \Drupal\Core\Config\StorableConfigBase
* The configuration object that caused the event to fire.
*/
public function getConfig() {
......
......@@ -140,8 +140,13 @@ final class ConfigEvents {
* @see \Drupal\Core\Config\ConfigFactoryOverrideBase
*
* @var string
*
* @deprecated in drupal:10.3.0 and is removed from drupal:11.0.0. Use
* \Drupal\Core\Config\ConfigCollectionEvents::COLLECTION_INFO instead.
*
* @see https://www.drupal.org/node/3406105
*/
const COLLECTION_INFO = 'config.collection_info';
const COLLECTION_INFO = ConfigCollectionEvents::COLLECTION_INFO;
/**
* Name of the event fired just before importing configuration.
......
......@@ -260,7 +260,8 @@ public function rename($old_name, $new_name) {
// Prime the cache and load the configuration with the correct overrides.
$config = $this->get($new_name);
$this->eventDispatcher->dispatch(new ConfigRenameEvent($config, $old_name), ConfigEvents::RENAME);
$event_name = $this->storage->getCollectionName() === StorageInterface::DEFAULT_COLLECTION ? ConfigEvents::RENAME : ConfigCollectionEvents::RENAME_IN_COLLECTION;
$this->eventDispatcher->dispatch(new ConfigRenameEvent($config, $old_name), $event_name);
return $this;
}
......
......@@ -10,7 +10,7 @@
abstract class ConfigFactoryOverrideBase implements EventSubscriberInterface {
/**
* Reacts to the ConfigEvents::COLLECTION_INFO event.
* Reacts to the ConfigCollectionEvents::COLLECTION_INFO event.
*
* @param \Drupal\Core\Config\ConfigCollectionInfo $collection_info
* The configuration collection info event.
......@@ -45,7 +45,7 @@ abstract public function onConfigRename(ConfigRenameEvent $event);
* {@inheritdoc}
*/
public static function getSubscribedEvents(): array {
$events[ConfigEvents::COLLECTION_INFO][] = ['addCollections'];
$events[ConfigCollectionEvents::COLLECTION_INFO][] = ['addCollections'];
$events[ConfigEvents::SAVE][] = ['onConfigSave', 20];
$events[ConfigEvents::DELETE][] = ['onConfigDelete', 20];
$events[ConfigEvents::RENAME][] = ['onConfigRename', 20];
......
......@@ -34,22 +34,32 @@ public function getCacheSuffix();
* it can have its own implementation of
* \Drupal\Core\Config\StorableConfigBase. Configuration overriders can link
* themselves to a configuration collection by listening to the
* \Drupal\Core\Config\ConfigEvents::COLLECTION_INFO event and adding the
* collections they are responsible for. Doing this will allow installation
* and synchronization to use the overrider's implementation of
* StorableConfigBase.
* \Drupal\Core\Config\ConfigCollectionEvents::COLLECTION_INFO event and
* adding the collections they are responsible for. Doing this will allow
* installation and synchronization to use the overrider's implementation of
* StorableConfigBase. Additionally, the overrider's implementation should
* trigger the appropriate event:
* - Saving and creating triggers ConfigCollectionEvents::SAVE_IN_COLLECTION.
* - Deleting triggers ConfigCollectionEvents::DELETE_IN_COLLECTION.
* - Renaming triggers ConfigCollectionEvents::RENAME_IN_COLLECTION.
*
* @see \Drupal\Core\Config\ConfigCollectionInfo
* @see \Drupal\Core\Config\ConfigImporter::importConfig()
* @see \Drupal\Core\Config\ConfigInstaller::createConfiguration()
* @see \Drupal\Core\Config\ConfigCollectionEvents::SAVE_IN_COLLECTION
* @see \Drupal\Core\Config\ConfigCollectionEvents::DELETE_IN_COLLECTION
* @see \Drupal\Core\Config\ConfigCollectionEvents::RENAME_IN_COLLECTION
*
* @param string $name
* The configuration object name.
* @param string $collection
* The configuration collection.
*
* @return \Drupal\Core\Config\StorableConfigBase
* The configuration object for the provided name and collection.
* @return \Drupal\Core\Config\StorableConfigBase|null
* The configuration object for the provided name and collection. NULL
* should be returned when the overrider does not use configuration
* collections. For example: a module that provides an overrider to avoid
* storing API keys in config would not use collections.
*/
public function createConfigObject($name, $collection = StorageInterface::DEFAULT_COLLECTION);
......
......@@ -220,7 +220,17 @@ public function uninstall($type, $name) {
// Remove any matching configuration from collections.
foreach ($this->activeStorage->getAllCollectionNames() as $collection) {
$collection_storage = $this->activeStorage->createCollection($collection);
$collection_storage->deleteAll($name . '.');
$overrider = $this->getConfigCollectionInfo()->getOverrideService($collection);
foreach ($collection_storage->listAll($name . '.') as $config_name) {
if ($overrider) {
$config = $overrider->createConfigObject($config_name, $collection);
}
else {
$config = new Config($config_name, $collection_storage, $this->eventDispatcher, $this->typedConfigManager);
}
$config->initWithData($collection_storage->read($config_name));
$config->delete();
}
}
$schema_dir = $this->extensionPathResolver->getPath($type, $name) . '/' . InstallStorage::CONFIG_SCHEMA_DIRECTORY;
......@@ -391,7 +401,7 @@ public function getConfigEntitiesToChangeOnDependencyRemoval($type, array $names
public function getConfigCollectionInfo() {
if (!isset($this->configCollectionInfo)) {
$this->configCollectionInfo = new ConfigCollectionInfo();
$this->eventDispatcher->dispatch($this->configCollectionInfo, ConfigEvents::COLLECTION_INFO);
$this->eventDispatcher->dispatch($this->configCollectionInfo, ConfigCollectionEvents::COLLECTION_INFO);
}
return $this->configCollectionInfo;
}
......
......@@ -17,12 +17,12 @@ class ConfigRenameEvent extends ConfigCrudEvent {
/**
* Constructs the config rename event.
*
* @param \Drupal\Core\Config\Config $config
* @param \Drupal\Core\Config\StorableConfigBase $config
* The configuration that has been renamed.
* @param string $old_name
* The old configuration object name.
*/
public function __construct(Config $config, $old_name) {
public function __construct(StorableConfigBase $config, $old_name) {
$this->config = $config;
$this->oldName = $old_name;
}
......
......@@ -2,6 +2,7 @@
namespace Drupal\Core\Config;
use Drupal\Component\Utility\NestedArray;
use Drupal\Core\Config\Schema\Ignore;
use Drupal\Core\Config\Schema\Mapping;
use Drupal\Core\Config\Schema\Sequence;
......@@ -121,6 +122,47 @@ public function getStorage() {
return $this->storage;
}
/**
* Gets original data from this configuration object.
*
* Original data is the data as it is immediately after loading from
* configuration storage before any changes. If this is a new configuration
* object it will be an empty array.
*
* @see \Drupal\Core\Config\Config::get()
*
* @param string $key
* A string that maps to a key within the configuration data.
*
* @return mixed
* The data that was requested.
*/
public function getOriginal($key = '') {
$original_data = $this->originalData;
if (empty($key)) {
return $original_data;
}
$parts = explode('.', $key);
if (count($parts) == 1) {
return $original_data[$key] ?? NULL;
}
$value = NestedArray::getValue($original_data, $parts, $key_exists);
return $key_exists ? $value : NULL;
}
/**
* Gets the raw data without any manipulations.
*
* @return array
* The raw data.
*/
public function getRawData() {
return $this->data;
}
/**
* Gets the schema wrapper for the whole configuration object.
*
......
......@@ -7,6 +7,11 @@
*
* Classes implementing this interface allow reading and writing configuration
* data from and to the storage.
*
* Note: this should never be used directly to work with active configuration.
* The values returned from it do not have the expected overrides and writing
* directly to the storage does not trigger configuration events. Use the
* 'config.factory' service and the configuration objects it provides.
*/
interface StorageInterface {
......
services:
config_events_test.event_subscriber:
config_collection_install_test.event_subscriber:
class: Drupal\config_collection_install_test\EventSubscriber
arguments: ['@state']
tags:
......
......@@ -2,8 +2,8 @@
namespace Drupal\config_collection_install_test;
use Drupal\Core\Config\ConfigCollectionEvents;
use Drupal\Core\Config\ConfigCollectionInfo;
use Drupal\Core\Config\ConfigEvents;
use Drupal\Core\State\StateInterface;
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
......@@ -27,7 +27,7 @@ public function __construct(StateInterface $state) {
}
/**
* Reacts to the ConfigEvents::COLLECTION_INFO event.
* Reacts to the ConfigCollectionEvents::COLLECTION_INFO event.
*
* @param \Drupal\Core\Config\ConfigCollectionInfo $collection_info
* The configuration collection info event.
......@@ -43,7 +43,7 @@ public function addCollections(ConfigCollectionInfo $collection_info) {
* {@inheritdoc}
*/
public static function getSubscribedEvents(): array {
$events[ConfigEvents::COLLECTION_INFO][] = ['addCollections'];
$events[ConfigCollectionEvents::COLLECTION_INFO][] = ['addCollections'];
return $events;
}
......
......@@ -2,6 +2,7 @@
namespace Drupal\config_events_test;
use Drupal\Core\Config\ConfigCollectionEvents;
use Drupal\Core\Config\ConfigCrudEvent;
use Drupal\Core\Config\ConfigEvents;
use Drupal\Core\State\StateInterface;
......@@ -31,17 +32,24 @@ public function __construct(StateInterface $state) {
*
* @param \Drupal\Core\Config\ConfigCrudEvent $event
* The configuration event.
* @param string $name
* @param string $event_name
* The event name.
*/
public function configEventRecorder(ConfigCrudEvent $event, $name) {
public function configEventRecorder(ConfigCrudEvent $event, $event_name) {
$config = $event->getConfig();
$this->state->set('config_events_test.event', [
'event_name' => $name,
$event_info = [
'event_name' => $event_name,
'current_config_data' => $config->get(),
'original_config_data' => $config->getOriginal(),
'raw_config_data' => $config->getRawData(),
]);
];
$this->state->set('config_events_test.event', $event_info);
// Record all events that occur.
$all_events = $this->state->get('config_events_test.all_events', []);
$config_name = $config->getName();
$all_events[$event_name][$config_name][] = $event_info;
$this->state->set('config_events_test.all_events', $all_events);
}
/**
......@@ -51,6 +59,9 @@ public static function getSubscribedEvents(): array {
$events[ConfigEvents::SAVE][] = ['configEventRecorder'];
$events[ConfigEvents::DELETE][] = ['configEventRecorder'];
$events[ConfigEvents::RENAME][] = ['configEventRecorder'];
$events[ConfigCollectionEvents::SAVE_IN_COLLECTION][] = ['configEventRecorder'];
$events[ConfigCollectionEvents::DELETE_IN_COLLECTION][] = ['configEventRecorder'];
$events[ConfigCollectionEvents::RENAME_IN_COLLECTION][] = ['configEventRecorder'];
return $events;
}
......
......@@ -3,6 +3,8 @@
namespace Drupal\language\Config;
use Drupal\Core\Cache\Cache;
use Drupal\Core\Config\ConfigCollectionEvents;
use Drupal\Core\Config\ConfigCrudEvent;
use Drupal\Core\Config\StorableConfigBase;
use Drupal\Core\Config\StorageInterface;
use Drupal\Core\Config\TypedConfigManagerInterface;
......@@ -62,6 +64,11 @@ public function save($has_trusted_data = FALSE) {
// an update of configuration, but only for a specific language.
Cache::invalidateTags($this->getCacheTags());
$this->isNew = FALSE;
// Dispatch configuration override event as detailed in
// \Drupal\Core\Config\ConfigFactoryOverrideInterface::createConfigObject().
$this->eventDispatcher->dispatch(new ConfigCrudEvent($this), ConfigCollectionEvents::SAVE_IN_COLLECTION);
// Dispatch an event specifically for language configuration override
// changes.
$this->eventDispatcher->dispatch(new LanguageConfigOverrideCrudEvent($this), LanguageConfigOverrideEvents::SAVE_OVERRIDE);
$this->originalData = $this->data;
return $this;
......@@ -75,6 +82,11 @@ public function delete() {
$this->storage->delete($this->name);
Cache::invalidateTags($this->getCacheTags());
$this->isNew = TRUE;
// Dispatch configuration override event as detailed in
// \Drupal\Core\Config\ConfigFactoryOverrideInterface::createConfigObject().
$this->eventDispatcher->dispatch(new ConfigCrudEvent($this), ConfigCollectionEvents::DELETE_IN_COLLECTION);
// Dispatch an event specifically for language configuration override
// changes.
$this->eventDispatcher->dispatch(new LanguageConfigOverrideCrudEvent($this), LanguageConfigOverrideEvents::DELETE_OVERRIDE);
$this->originalData = $this->data;
return $this;
......
config_events_test.test:
type: config_object
label: 'Configuration events test'
mapping:
key:
type: string
label: 'Value'
name: 'Language events test'
type: module
package: Testing
version: VERSION
services:
language_events_test.event_subscriber:
class: Drupal\language_events_test\EventSubscriber
arguments: ['@state']
tags:
- { name: event_subscriber }
<?php
namespace Drupal\language_events_test;
use Drupal\Core\State\StateInterface;
use Drupal\language\Config\LanguageConfigOverrideEvents;
use Drupal\language\Config\LanguageConfigOverrideCrudEvent;
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
class EventSubscriber implements EventSubscriberInterface {
/**
* Constructs the Event Subscriber object.
*
* @param \Drupal\Core\State\StateInterface $state
* The state key value store.
*/
public function __construct(private StateInterface $state) {
}
/**
* Reacts to config event.
*
* @param \Drupal\language\Config\LanguageConfigOverrideCrudEvent $event
* The language configuration event.
* @param string $event_name
* The event name.
*/
public function configEventRecorder(LanguageConfigOverrideCrudEvent $event, string $event_name): void {
$override = $event->getLanguageConfigOverride();
$event_info = [
'event_name' => $event_name,
'current_override_data' => $override->get(),
'original_override_data' => $override->getOriginal(),
];
// Record all events that occur.
$all_events = $this->state->get('language_events_test.all_events', []);
$override_name = $override->getName();
if (!isset($all_events[$event_name][$override_name])) {
$all_events[$event_name][$override_name] = [];
}
$all_events[$event_name][$override_name][] = $event_info;
$this->state->set('language_events_test.all_events', $all_events);
}
/**
* {@inheritdoc}
*/
public static function getSubscribedEvents(): array {
$events[LanguageConfigOverrideEvents::SAVE_OVERRIDE][] = ['configEventRecorder'];
$events[LanguageConfigOverrideEvents::DELETE_OVERRIDE][] = ['configEventRecorder'];
return $events;
}
}
......@@ -2,6 +2,7 @@
namespace Drupal\Tests\language\Functional;
use Drupal\Core\Config\ConfigCollectionEvents;
use Drupal\language\Entity\ConfigurableLanguage;
use Drupal\Tests\BrowserTestBase;
......@@ -91,7 +92,12 @@ public function testConfigOverrideImportEvents() {
// Test that no config save event has been fired during the import because
// language configuration overrides do not fire events.
$event_recorder = \Drupal::state()->get('config_events_test.event', FALSE);
$this->assertFalse($event_recorder);
$this->assertSame([
'event_name' => ConfigCollectionEvents::SAVE_IN_COLLECTION,
'current_config_data' => ['name' => 'FR default site name'],
'original_config_data' => [],
'raw_config_data' => ['name' => 'FR default site name'],
], $event_recorder);
$this->drupalGet('fr');
$this->assertSession()->pageTextContains('FR default site name');
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment