Skip to content

Configuration management

Configuration is discovered and loaded at the time of command execution. So, it is not available to be used during the bootstrap phase.

Contexts

Polymer's configuration system discovers and loads contexts provided by Core and it's extensions. The contexts are layered atop one and other, where contexts on top have the highest priority when retrieving a configuration value.

Example

Polymer discovers and loads, in order, the following contexts:

  • system
  • polymer
  • extension_a
  • extension_b
  • project
  • other
  • process

The polymer and project contexts both specify the configuration value keyed by my.foo. polymer specifies it as my.foo=bar and project as my.foo=baz.

When the value my.foo is retrieved during runtime, the value baz is returned.

The process context

The process context is a special context that is always included at the top of the context stack. Any configuration that is set during runtime is stored in this context. That means any configuration set during runtime will override contexts with equivalent keys.

Discovering contexts

As the command is about to execute, Polymer initiates configuration compilation by discovering all available contexts, altering them, and finally compiling them, at which point the configuration is ready to be used.

Registered event subscribers can subscribe to the following:

  • CollectConfigContextsEvent::class
  • AlterConfigContextsEvent::class

For example, in an extension's service provider:

public function register(): void
{
    $container = $this->getContainer();
    $container->addShared('drupalConfigContextProvider', ContextProvidersSubscriber::class)
        ->addArgument(new ResolvableArgument('drupalFileSystem'));
}

public function boot(): void
{
    $container = $this->getContainer();
    $container->extend('eventDispatcher')
        ->addMethodCall('addSubscriber', ['drupalConfigContextProvider']);
}

And in ContextProvidersSubscriber:

public function addContexts(CollectConfigContextsEvent $event): void
{
    try {
        $this->drupalFileSystem->getDrupalRoot();
    } catch (\OutOfBoundsException $e) {
        // If Drupal root is not found, skip adding environment configuration.
        return;
    }
    $site = $event->getInput()->getOption('site');
    $environment = $event->getInput()->getOption('environment');
    $drupalConfig = [];
    $possibleConfigFiles = [];
    if (is_string($site) && in_array($site, $this->drupalFileSystem->getMultisiteDirs())) {
        $sitePath = $this->drupalFileSystem->getDrupalRoot() . '/sites/' . $site;
        $possibleConfigFiles['site'] =  $sitePath . '/polymer.yml';
        if (is_string($environment)) {
            $possibleConfigFiles['site_environment'] = $sitePath . '/' . $environment . '.polymer.yml';
        }
    }
    $possibleConfigFiles = array_filter($possibleConfigFiles, function ($file) {
        return file_exists($file);
    });
    foreach ($possibleConfigFiles as $configId => $file) {
        $loader = new YamlConfigLoader();
        $drupalConfig[$configId] = $loader->load($file)->export();
    }
    $event->addContexts($drupalConfig);
}

public static function getSubscribedEvents(): array
{
    $events = [
        CollectConfigContextsEvent::class => [
            ['addContexts', -1000]
        ],
    ];
    return $events;
}

The above will add the site and site_environment contexts to the configuration.

Tokenized values

Configuration values can be tokenized. For example, a context has foo.bar: ${my.config.key}. Assuming my.config.key can be resolved, foo.bar will be set to the value of my.config.key.