ソースコードには含めたくない設定内容を外部ファイルから読み込みたい、というケースはあらゆるアプリケーションで発生します。
今までも、各種 PHP スクリプトで活用できる Config ツールを書いてきており、その内容は単一の config.json ファイルを読み込み、すべての設定をスタティックで保持し、値もスタティック関数で読み込む、という単純なものでした。
それらは今までのアプリケーションによく適応できましたが、アプリケーション規模が大きくなるにつれ、単一の設定ファイルだけのサポートでは見通しが悪くなってきました。
そこで、以下の様なコンフィグツールを設計してみます。
<?php namespace Wws\Config; use Symfony\Component\Yaml\Yaml; use Yosymfony\Toml\Toml; class Config { private static $instances = []; public static function loadDir(string $dir): void { $files = scandir($dir); foreach ($files as $file) { if (in_array(['.', '..'], $file)) continue; $ns = pathinfo($file, PATHINFO_FILENAME); $path = "{$dir}/{$file}"; Config::load($ns, $path); } } public static function load(string $ns, string $path): void { $instance = new static(); $instance->loadFile($path); Config::setInstance($ns, $instance); } public static function setInstance(string $ns, Config $instance): void { self::$instances[$ns] = $instance; } public static function getInstance(string $ns, Config $instance): Config { return self::$instances[$ns]; } const INI = 'ini'; const YML = 'yml'; const YAML = 'yaml'; const JSON = 'json'; const TOML = 'toml'; private $_data = null; public function loadFile(string $path): void { if (!file_exists($path)) { throw new \InvalidArgumentException($path . ' not found. Please specify existing file.'); } if (!$file = file_get_contents($path)) { throw new \InvalidArgumentException($path . ' could not read. Please confirm access rights.'); }; if (self::YAML === substr($path, -1 * strlen(self::YAML)) || $type == self::YAML) { $this->_data = Yaml::parse($file); } elseif (self::YML === substr($path, -1 * strlen(self::YML)) || $type == self::YML) { $this->_data = Yaml::parse($file); } elseif (self::JSON === substr($path,-1 * strlen(self::JSON)) || $type == self::JSON) { $this->_data = json_decode($file, true); } elseif (self::INI === substr($path,-1 * strlen(self::INI)) || $type == self::INI) { $this->_data = Toml::Parse($file); } elseif (self::TOML === substr($path,-1 * strlen(self::TOML)) || $type == self::TOML) { $this->_data = Toml::Parse($file); } else { throw new \InvalidArgumentException('Please specify valid config file type. .yaml .json .toml'); } } public function setValue(string $path, $obj, string $delimiter='.') { $nodes = explode($delimiter, $path); $value = &$this->_data; for ($i=0; $i < count($nodes); $i++) { $node = $nodes[$i]; if ($i == count($nodes) - 1) { if (preg_match('/(?P<prop>\w+)\[(?P<idx>\d*)\]/', $node, $matches)) { $prop = $matches['prop']; if (!array_key_exists($prop, $value)) $value[$prop] = []; if ('' === $matches['idx']) { $value[$prop][] = $obj; } else { $idx = (int)$matches['idx']; $value[$prop][$idx] = $obj; } } else { $value[$node] = $obj; } } else { if (preg_match('/(?P<prop>\w+)\[(?P<idx>\d+)\]/', $node, $matches)) { $prop = $matches['prop']; if (!array_key_exists($prop, $value)) $value[$prop] = []; if ('' === $matches['idx']) { $len = count($value[$prop]); $value[$prop][] = []; $value = &$value[$prop][$len]; } else { $idx = (int)$matches['idx']; $value = &$value[$prop][$idx]; } } else { $value = &$value[$node]; } } } } public function hasValue(?string $path=null, string $delimiter='.') { if (is_null($path)) return true; try { $this->getValue($path, $delimiter); return true; } catch (\OutOfRangeException $e) { return false; } } public function getValue(?string $path=null, string $delimiter='.') { if (is_null($path)) return $this->_data; $nodes = explode($delimiter, $path); $value = $this->_data; foreach ($nodes as $node) { if ($node === '') continue; if (preg_match('/(?P<prop>\w+)\[(?P<idx>\d*)\]/', $node, $matches)) { $prop = $matches['prop']; $idx = (int)$matches['idx']; if (!array_key_exists($prop, $value) || !array_key_exists($idx, $value[$prop])) { throw new \OutOfRangeException('The config has no value for '.$path); } $value = $value[$prop][$idx]; } else { if (!array_key_exists($node, $value)) { throw new \OutOfRangeException('The config has no value for '.$path); } $value = $value[$node]; } } return $value; } }