illuminatech / config
Provides support for Laravel application runtime configuration managed in persistent storage
Fund package maintenance!
klimov-paul
Patreon
Installs: 18 530
Dependents: 1
Suggesters: 0
Security: 0
Stars: 150
Watchers: 6
Forks: 14
Open Issues: 0
Requires
- illuminate/support: ^9.0 || ^10.0 || ^11.0
Requires (Dev)
- illuminate/cache: *
- illuminate/config: *
- illuminate/database: *
- illuminate/encryption: *
- illuminate/validation: *
- phpunit/phpunit: ^8.0 || ^9.3 || ^10.5
- psr/log: ^1.0
README
Laravel Persistent Configuration Repository
This extension introduces persistent configuration repository for Laravel. Its usage in particular provides support for application runtime configuration, loading config from database.
For license information check the LICENSE-file.
Installation
The preferred way to install this extension is through composer.
Either run
php composer.phar require --prefer-dist illuminatech/config
or add
"illuminatech/config": "*"
to the require section of your composer.json.
Usage
This extension allows reconfiguration of already created config repository using data from the external storage like relational database.
It provides special config repository class \Illuminatech\Config\PersistentRepository
, which wraps any given config repository,
adding a layer for saving and restoring of data from the persistent storage.
<?php use Illuminate\Config\Repository; use Illuminate\Support\Facades\App; use Illuminatech\Config\PersistentRepository; use Illuminatech\Config\StorageDb; $sourceConfigRepository = new Repository([ 'foo' => [ 'name' => 'Foo', ], 'bar' => [ 'enabled' => false, ], 'other' => [ 'value' => 'Some', ], ]); $storage = new StorageDb(App::make('db.connection')); $persistentConfigRepository = (new PersistentRepository($sourceConfigRepository, $storage)) ->setItems([ 'foo.name', 'bar.enabled', ]); echo $persistentConfigRepository->get('foo.name'); // returns value from database if present, otherwise the one from source repository, in this case - 'Foo' echo $persistentConfigRepository->get('other.value'); // keys, which are not specified as "items" always remain intact, in this case - always return 'Some'
Config data, which should be saved in persistent storage defined via \Illuminatech\Config\PersistentRepository::setItems()
.
Only keys, which are explicitly defined as "items", will be stored or retrieved from the persistent storage. Any other data
present in the source config repository will remain as it is.
PersistentRepository fully decorates any config repository, matching \Illuminate\Contracts\Config\Repository
and can substitute \Illuminate\Config\Repository
instance.
In particular this allows you to substitute regular Laravel config by \Illuminatech\Config\PersistentRepository
instance,
applying configuration from database to the entire application. You can do so in your AppServiceProvider
class. For example:
<?php namespace App\Providers; use Illuminate\Contracts\Config\Repository; use Illuminate\Support\ServiceProvider; use Illuminatech\Config\PersistentRepository; use Illuminatech\Config\StorageDb; class AppServiceProvider extends ServiceProvider { public function boot() { $this->app->extend('config', function (Repository $originConfig) { $storage = new StorageDb($this->app->make('db.connection')); $newConfig = (new PersistentRepository($originConfig, $storage)) ->setItems([ 'mail.contact.address' => [ 'label' => __('Email address receiving contact messages'), 'rules' => ['sometimes', 'required', 'email'], ], // ... ]); return $newConfig; }); // ... } }
Then anytime you access 'config' service in your application via config()
function, \Illuminate\Support\Facades\Config
facade
or via service container you will interact with \Illuminatech\Config\PersistentRepository
instance getting values modified
by database data.
Tip: with this extension there is no need for manually putting any data into actual storage (e.g. writing DB migrations or Seeding) - it will be filled up automatically. If value for the particular item is missing in the storage, it will be simply picked up from the wrapped config repository (e.g. as defined at 'config/*.php' files). However, some storages require some preparations before they can function, like creating a database table.
Note: this extension does not provide built in service provider for application config substitute as it might be not desired
for particular application, while \Illuminatech\Config\PersistentRepository
usage is not limited with this task.
However, you can use \Illuminatech\Config\Providers\AbstractPersistentConfigServiceProvider
class as a scaffold for such service provider.
For example:
<?php namespace App\Providers; use Illuminatech\Config\Providers\AbstractPersistentConfigServiceProvider; use Illuminatech\Config\StorageContract; use Illuminatech\Config\StorageDb; class PersistentConfigServiceProvider extends AbstractPersistentConfigServiceProvider { protected function storage(): StorageContract { return (new StorageDb($this->app->make('db.connection'))); } protected function items(): array { return [ 'mail.contact.address' => [ 'label' => __('Email address receiving contact messages'), 'rules' => ['sometimes', 'required', 'email'], ], // ... ]; } }
Do not forget to register your particular persistent config service provider in "providers" section at "config/app.php":
<?php return [ // ... 'providers' => [ // ... App\Providers\PersistentConfigServiceProvider::class, ], // ... ];
You may also manage persistent configuration per particular application entity. For example: imagine we need to allow
application user to customize appearance of his profile page, like changing color schema or enable/disable sidebar and so on.
Such settings can be managed by \Illuminatech\Config\PersistentRepository
bound to the user Eloquent model. Such model class
may look like following:
<?php namespace App\Models; use Illuminate\Config\Repository; use Illuminate\Database\Eloquent\Model; use Illuminatech\Config\PersistentRepository; use Illuminatech\Config\StorageDb; class User extends Model { /** * @var \Illuminatech\Config\PersistentRepository configuration repository specific to this model. */ private $config; /** * Returns configuration associated with this particular model. * * @return \Illuminatech\Config\PersistentRepository config repository. */ public function getConfig(): PersistentRepository { if ($this->config === null) { if (empty($this->id)) { throw new \InvalidArgumentException('Unable to get config for model without ID.'); } $repository = new Repository($this->defaultConfigData()); $storage = (new StorageDb($this->getConnection())) ->setFilter(['user_id' => $this->id]); // ensure configuration varies per each model $this->config = (new PersistentRepository($repository, $storage)) ->setItems($this->persistentConfigItems()); } return $this->config; } /** * Defines default configuration for the model instance. * * @return array config. */ private function defaultConfigData() { return [ 'sidebar' => [ 'enabled' => true, ], 'color' => [ 'primary' => '#4099de', 'sidebar' => '#b3c1d1', ], ]; } /** * Defines the config items, which should be manageable from web interface and stored in the database. * * @return array config items. */ private function persistentConfigItems(): array { return [ 'sidebar.enabled' => [ 'label' => 'Sidebar enabled', 'rules' => ['sometimes', 'required', 'boolean'], ], 'color.primary' => [ 'label' => 'Primary color', 'rules' => ['sometimes', 'required', 'string'], ], 'color.sidebar' => [ 'label' => 'Sidebar color', 'rules' => ['sometimes', 'required', 'string'], ], ]; } }
It will allow you to operate persistent configuration per each user record separately, so profile page composition may look like following:
@php /* @var $user App\Models\User */ @endphp @extends('layouts.main') @section('content') @if ($user->getConfig()->get('sidebar.enabled')) @include('includes.sidebar', ['color' => $user->getConfig()->get('color.sidebar')]) @endif <div style="background-color:{{ $user->getConfig()->get('color.primary') }};"> ... </div> @endsection
Configuration items specification
Config parts, which should be saved in the persistent storage are defined by \Illuminatech\Config\PersistentRepository::setItems()
,
which accepts a list of \Illuminatech\Config\Item
or configuration array for it.
Each configuration item should define a key, which leads to the target value in source repository.
Configuration item also has several properties, which supports creation of web interface for configuration setup.
These are:
- 'id' - string, item unique ID in the list, this value will be used in request fields and form inputs.
- 'label' - string, verbose label for the config value input.
- 'hint' - string, verbose description for the config value or input hint.
- 'rules' - array, value validation rules.
- 'cast' - string, native type for the value to be cast to.
- 'encrypt' - bool, whether to encrypt value for the storage or not.
- 'options' - array, additional descriptive options for the item, which can be used as you see fit.
Since only 'key' is mandatory item may be specified by single string defining this key.
Here are some examples of item specifications:
<?php use Illuminatech\Config\Item; use Illuminatech\Config\PersistentRepository; $persistentConfigRepository = (new PersistentRepository(...)) ->setItems([ 'some.config.value', 'another.config.value' => [ 'label' => 'Custom label', 'rules' => ['required', 'numeric'], ], [ 'key' => 'array.config.value', 'rules' => ['required', 'array'], 'cast' => 'array', ], new Item(['key' => 'explicit.object']), ]);
Configuration storage
Declared configuration items may be saved into persistent storage and then retrieved from it.
The actual item storage can be any class matching \Illuminatech\Config\StorageContract
interface.
Following storages are available within this extension:
- \Illuminatech\Config\StorageDb - stores configuration inside relational database;
- \Illuminatech\Config\StorageEloquent - stores configuration using Eloquent models;
- \Illuminatech\Config\StoragePhp - stores configuration in local PHP files;
- \Illuminatech\Config\StorageArray - stores configuration in runtime memory;
Please refer to the particular storage class for more details.
Saving and restoring data
\Illuminatech\Config\PersistentRepository
will automatically retrieve config item values from persistent storage on the
first attempt to get config value from it.
<?php use Illuminatech\Config\PersistentRepository; $persistentConfigRepository = (new PersistentRepository(...)) ->setItems([ 'some.config', ]); $value = $persistentConfigRepository->get('some.config'); // loads data from persistent storage automatically.
You may also manually fetch data from persistent storage using restore()
method:
<?php use Illuminatech\Config\PersistentRepository; $persistentConfigRepository = (new PersistentRepository(...)) ->setItems([ 'some.config', ]); $persistentConfigRepository->restore(); // loads/re-loads data from persistent storage
Heads up! Any error or exception, which appears during values restoration, will be automatically suppressed. This is done to avoid application blocking in case storage is not yet ready for usage, for example: database table does not yet exist. Storage failure error will appear only at the application log. You should manually test value restoration is working at your application to avoid unexpected behavior.
To save config data into persistent storage use method save()
:
<?php use Illuminatech\Config\PersistentRepository; $persistentConfigRepository = (new PersistentRepository(...)) ->setItems([ 'some.config', 'another.config', ]); $persistentConfigRepository->save([ 'some.config' => 'some persistent value', 'another.config' => 'another persistent value', ]);
Changes made via regular config repository interface (e.g. via methods set()
, push()
and so on) will not be automatically
saved into the persistent storage. However, you may use synchronize()
method to save current config item values into it.
<?php use Illuminatech\Config\PersistentRepository; $persistentConfigRepository = (new PersistentRepository(...)) ->setItems([ 'some.config', 'another.config', ]); $persistentConfigRepository->set('some.config', 'new value'); // no changes at the persistent storage at this point $persistentConfigRepository->synchronize(); // save values to the persistent storage
Tip: You may invoke
synchronize()
at the application terminating stage ensuring all changes made during application running are saved.
Method reset()
clears all data saved to the persistent storage, restoring original (e.g. default) config repository values.
<?php use Illuminate\Config\Repository; use Illuminatech\Config\PersistentRepository; $sourceConfigRepository = new Repository([ 'some' => [ 'config' => 'original value', ], ]); $persistentConfigRepository = (new PersistentRepository($sourceConfigRepository, ...)) ->setItems([ 'some.config', ]); $persistentConfigRepository->save([ 'some.config' => 'new value', ]); echo $persistentConfigRepository->get('some.config'); // outputs 'new value' $persistentConfigRepository->reset(); // clears data in the persistent storage echo $persistentConfigRepository->get('some.config'); // outputs 'original value'
You can also use resetValue()
method to reset particular config key only.
Caching
You can use PSR-16 compatible cache storage to improve performance of the config item retrieval from persistent storage. For example:
<?php use Illuminate\Config\Repository; use Illuminate\Support\Facades\App; use Illuminatech\Config\PersistentRepository; $sourceConfigRepository = new Repository([ 'some' => [ 'config' => 'original value', ], ]); $persistentConfigRepository = (new PersistentRepository($sourceConfigRepository, ...)) ->setItems([ 'some.config', ]) ->setCache(App::make('cache.store')) ->setCacheKey('global-config') ->setCacheTtl(3600 * 24);
Validation
Each configuration item comes with validation rules, which matches ['sometimes' ,'required']
by default. You can easily
create a validation for the user input before config saving, using these rules, or use \Illuminatech\Config\PersistentRepository::validate()
.
For example:
<?php /* @var $request Illuminate\Http\Request */ /* @var $config Illuminatech\Config\PersistentRepository */ $validatedData = $config->validate($request->all()); // throws \Illuminate\Validation\ValidationException if validation fails. // ...
You can also use \Illuminatech\Config\PersistentRepository::makeValidator()
method to create a validator instance for manual processing.
Heads up! Watch for usage dot symbols ('.') inside the input in case you do not use \Illuminatech\Config\PersistentRepository::validate()
method.
By default Laravel considers dots in validation rules as array nested keys separator. You should either prefix them
with backslash ('\') or manually define \Illuminatech\Config\Item::$id
in the way it does not contain a dot.
Creating configuration web interface
One of the most common use case for this extension is creating a web interface, which allows control of application
configuration in runtime.
\Illuminatech\Config\PersistentRepository
serves not only for applying of the configuration - it also helps to create an
interface for configuration editing.
The web controller for configuration management may look like following:
<?php namespace App\Http\Controllers; use Illuminate\Contracts\Container\Container; use Illuminate\Http\Request; class ConfigController extends Controller { /** * @var \Illuminatech\Config\PersistentRepository persistent config repository, which is set at `AppServiceProvider`. */ private $config; public function __construct(Container $app) { $this->config = $app->get('config'); } public function index() { $this->config->restore(); // ensure config values restored from database return view('config.form', ['items' => $this->config->getItems()]); } public function update(Request $request) { $validatedData = $this->config->validate($request->all()); $this->config->save($validatedData); return back()->with('status', 'success'); } public function restoreDefaults() { $this->config->reset(); return back()->with('status', 'success'); } }
You can operate \Illuminatech\Config\Item
interface during HTML form input composition. For example:
... <form ...> ... @foreach ($items as $item) <label>{{ $item->label }}</label> <input type="text" name="{{ $item->id }}" value="{{ $item->getValue() }}"> <p>{{ $item->hint }}</p> @endforeach ... </form> ...
Tip: you can use
\Illuminatech\Config\Item::$options
to setup configuration for the dynamic form inputs, specifying input type, CSS class and so on inside of it.
Heads up! Remember that PHP automatically replaces non-alphanumeric characters like dot ('.'), dash ('-') and so on
inside request keys during native 'POST' parsing, making collection and validation for keys like 'config.some-key' impossible.
You will need to setup \Illuminatech\Config\Item::$id
value for each persistent configuration item manually, in case you
going to submit values via regular 'POST' request. For example:
<?php use Illuminatech\Config\PersistentRepository; $persistentConfigRepository = (new PersistentRepository(...)) ->setItems([ 'some.config.value' => [ 'id' => 'some_config_value', ], 'another-config-value' => [ 'id' => 'another_config_value', ], // ... ]);
Tip: you will not face this problem in case you submit configuration item values via REST API interface using JSON format or via native (not spoofed) 'PUT' request.
In case you are using Laravel Nova for your application admin panel, you can easily create an application configuration setup interface with illuminatech/nova-config extension.
Typecast
You may operate complex type values like arrays as a persistent ones. In order to do so, you should specify config item
typecasting via \Illuminatech\Config\Item::$cast
. For example:
<?php use Illuminate\Config\Repository; use Illuminatech\Config\PersistentRepository; $sourceConfigRepository = new Repository([ 'some' => [ 'array' => ['one', 'two', 'three'], ], ]); $persistentConfigRepository = (new PersistentRepository($sourceConfigRepository, ...)) ->setItems([ 'some.array' => [ 'cast' => 'array', // cast value from persistent storage to array 'rules' => ['sometimes', 'required', 'array'], ], ]); $persistentConfigRepository->save([ 'some.array' => ['five', 'six'], ]); $persistentConfigRepository->restore(); var_dump($persistentConfigRepository->get('some.array') === ['five', 'six']); // outputs 'true'
Encryption
In case you are planning to operate sensitive data like passwords, API keys and so on, you may want to store them as an
encrypted strings rather than the plain ones. This can be achieved enabling \Illuminatech\Config\Item::$encrypt
.
For example:
<?php use Illuminate\Config\Repository; use Illuminatech\Config\PersistentRepository; $sourceConfigRepository = new Repository([ 'some' => [ 'apiKey' => 'secret', ], ]); $persistentConfigRepository = (new PersistentRepository($sourceConfigRepository, ...)) ->setItems([ 'some.apiKey' => [ 'encrypt' => true, // encrypt value before placing it into the persistent storage ], ]);
Note that data encryption will impact the config repository performance.
Garbage collection
As your project evolves new configuration items may appear as well as some becomes redundant.
\Illuminatech\Config\PersistentRepository
automatically ignores any value in persistent storage in case it has no
matching config item set by setItems()
. Thus stored obsolete values will not affect config repository anyway, however
they still may consume extra space inside the storage. You may manually remove all obsolete values from the storage,
using gc()
method:
<?php use Illuminate\Config\Repository; use Illuminatech\Config\PersistentRepository; use Illuminatech\Config\StorageDb; $sourceConfigRepository = new Repository([ 'some' => [ 'config' => 'original value', ], ]); $storage = new StorageDb(...); $storage->save([ 'some.config' => 'some value', 'obsolete.config' => 'obsolete value', ]); $persistentConfigRepository = (new PersistentRepository($sourceConfigRepository, $storage)) ->setItems([ 'some.config', ]); $persistentConfigRepository->gc(); // removes 'obsolete.config' from storage
In case Illuminatech\Config\PersistentRepository::$gcEnabled
enabled garbage collection will be performed automatically
each time config values are saved via save()
or synchronize()
method.