Files
FreshRSS/lib/Minz/Configuration.php
Alexandre Alapetite dbbae15a84 Remove ConfigurationSetter (#5251)
This class has not been maintained for a while. Only a subset of our configuration properties are there, and some code is not relevant anymore. Furthermore, it is relying exclusively on dynamically invoked functions, making it challenging to maintain, in particular to find out what is used and what is not, what is handled and what is not.
It is not well suited for changes in data formats, which have been handled in the Context class instead.
It is also not able to handle configuration properties that are missing.
It is the class with most errors for PHPStan level 6 (179 errors). It is also making intense use of is_callable and call_user_func_array, which are performance killers.
Should the need arrise again to perform validation of our internal configuration files, I suggest an implementation with the opposite approach, namely driven by our code instead of driven by the data.
In summary, at the moment, this class is costly, while not offering many guarantees.
2023-04-07 12:40:43 +02:00

221 lines
6.2 KiB
PHP

<?php
/**
* Manage configuration for the application.
* @property-read string $base_url
* @property array<string|array<int,string>> $db
* @property-read string $disable_update
* @property-read string $environment
* @property array<string,bool> $extensions_enabled
* @property-read string $mailer
* @property-read array<string|int|bool> $smtp
* @property string $title
*/
class Minz_Configuration {
/**
* The list of configurations.
* @var array<string,static>
*/
private static $config_list = array();
/**
* Add a new configuration to the list of configuration.
*
* @param string $namespace the name of the current configuration
* @param string $config_filename the filename of the configuration
* @param string $default_filename a filename containing default values for the configuration
* @param object $configuration_setter an optional helper to set values in configuration
*/
public static function register(string $namespace, string $config_filename, string $default_filename = null, object $configuration_setter = null): void {
self::$config_list[$namespace] = new static(
$namespace, $config_filename, $default_filename, $configuration_setter
);
}
/**
* Parse a file and return its data.
*
* @param string $filename the name of the file to parse.
* @return array<string,mixed> of values
* @throws Minz_FileNotExistException if the file does not exist or is invalid.
*/
public static function load(string $filename): array {
$data = @include($filename);
if (is_array($data)) {
return $data;
} else {
throw new Minz_FileNotExistException($filename);
}
}
/**
* Return the configuration related to a given namespace.
*
* @param string $namespace the name of the configuration to get.
* @return static object
* @throws Minz_ConfigurationNamespaceException if the namespace does not exist.
*/
public static function get(string $namespace) {
if (!isset(self::$config_list[$namespace])) {
throw new Minz_ConfigurationNamespaceException(
$namespace . ' namespace does not exist'
);
}
return self::$config_list[$namespace];
}
/**
* The namespace of the current configuration.
* Unused.
* @phpstan-ignore-next-line
*/
private $namespace = '';
/**
* The filename for the current configuration.
* @var string
*/
private $config_filename = '';
/**
* The filename for the current default values, null by default.
* @var string|null
*/
private $default_filename = null;
/**
* The configuration values, an empty array by default.
* @var array<string,mixed>
*/
private $data = array();
/**
* An object which help to set good values in configuration.
* @var object|null
*/
private $configuration_setter = null;
/**
* Create a new Minz_Configuration object.
*
* @param string $namespace the name of the current configuration.
* @param string $config_filename the file containing configuration values.
* @param string $default_filename the file containing default values, null by default.
* @param object $configuration_setter an optional helper to set values in configuration
*/
private final function __construct(string $namespace, string $config_filename, string $default_filename = null, object $configuration_setter = null) {
$this->namespace = $namespace;
$this->config_filename = $config_filename;
$this->default_filename = $default_filename;
$this->_configurationSetter($configuration_setter);
if ($this->default_filename != null) {
$this->data = self::load($this->default_filename);
}
try {
$this->data = array_replace_recursive(
$this->data, self::load($this->config_filename)
);
} catch (Minz_FileNotExistException $e) {
if ($this->default_filename == null) {
throw $e;
}
}
}
/**
* Set a configuration setter for the current configuration.
* @param object|null $configuration_setter the setter to call when modifying data. It
* must implement an handle($key, $value) method.
*/
public function _configurationSetter(?object $configuration_setter): void {
if (is_callable(array($configuration_setter, 'handle'))) {
$this->configuration_setter = $configuration_setter;
}
}
public function configurationSetter(): object {
return $this->configuration_setter;
}
/**
* Check if a parameter is defined in the configuration
*/
public function hasParam(string $key): bool {
return isset($this->data[$key]);
}
/**
* Return the value of the given param.
*
* @param string $key the name of the param.
* @param mixed $default default value to return if key does not exist.
* @return array|mixed value corresponding to the key.
* @throws Minz_ConfigurationParamException if the param does not exist
*/
public function param(string $key, $default = null) {
if (isset($this->data[$key])) {
return $this->data[$key];
} elseif (!is_null($default)) {
return $default;
} else {
Minz_Log::warning($key . ' does not exist in configuration');
return null;
}
}
/**
* A wrapper for param().
* @return array|mixed
*/
public function __get(string $key) {
return $this->param($key);
}
/**
* Set or remove a param.
*
* @param string $key the param name to set.
* @param mixed $value the value to set. If null, the key is removed from the configuration.
*/
public function _param(string $key, $value = null): void {
if (!is_null($this->configuration_setter) && $this->configuration_setter->support($key)) {
$this->configuration_setter->handle($this->data, $key, $value);
} elseif (isset($this->data[$key]) && is_null($value)) {
unset($this->data[$key]);
} elseif (!is_null($value)) {
$this->data[$key] = $value;
}
}
/**
* A wrapper for _param().
* @param mixed $value
*/
public function __set(string $key, $value): void {
$this->_param($key, $value);
}
/**
* Save the current configuration in the configuration file.
*/
public function save(): bool {
$back_filename = $this->config_filename . '.bak.php';
@rename($this->config_filename, $back_filename);
if (file_put_contents($this->config_filename,
"<?php\nreturn " . var_export($this->data, true) . ';', LOCK_EX) === false) {
return false;
}
// Clear PHP cache for include
if (function_exists('opcache_invalidate')) {
opcache_invalidate($this->config_filename);
}
return true;
}
}