Passed
Push — master ( ae3156...6fb22a )
by Esteban De La Fuente
03:24
created

Configuration::__construct()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 4
Code Lines 2

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 3
CRAP Score 1

Importance

Changes 1
Bugs 0 Features 0
Metric Value
cc 1
eloc 2
c 1
b 0
f 0
nc 1
nop 1
dl 0
loc 4
ccs 3
cts 3
cp 1
crap 1
rs 10
1
<?php
2
3
declare(strict_types=1);
4
5
/**
6
 * Derafu: Biblioteca PHP (Núcleo).
7
 * Copyright (C) Derafu <https://www.derafu.org>
8
 *
9
 * Este programa es software libre: usted puede redistribuirlo y/o modificarlo
10
 * bajo los términos de la Licencia Pública General Affero de GNU publicada por
11
 * la Fundación para el Software Libre, ya sea la versión 3 de la Licencia, o
12
 * (a su elección) cualquier versión posterior de la misma.
13
 *
14
 * Este programa se distribuye con la esperanza de que sea útil, pero SIN
15
 * GARANTÍA ALGUNA; ni siquiera la garantía implícita MERCANTIL o de APTITUD
16
 * PARA UN PROPÓSITO DETERMINADO. Consulte los detalles de la Licencia Pública
17
 * General Affero de GNU para obtener una información más detallada.
18
 *
19
 * Debería haber recibido una copia de la Licencia Pública General Affero de GNU
20
 * junto a este programa.
21
 *
22
 * En caso contrario, consulte <http://www.gnu.org/licenses/agpl.html>.
23
 */
24
25
namespace Derafu\Lib\Core\Foundation;
26
27
use Composer\InstalledVersions;
28
use Derafu\Lib\Core\Foundation\Contract\ConfigurationInterface;
29
use Derafu\Lib\Core\Support\Store\Contract\DataContainerInterface;
30
use Derafu\Lib\Core\Support\Store\DataContainer;
31
use Symfony\Component\Yaml\Yaml;
32
33
/**
34
 * Configuración de la aplicación.
35
 */
36
class Configuration implements ConfigurationInterface
37
{
38
    /**
39
     * Directorio raíz de la aplicación.
40
     */
41
    private string $projectDir;
42
43
    /**
44
     * Prefijo para las configuraciones de Derafu Lib en el archivo de
45
     * configuración.
46
     *
47
     * Este prefijo no es necesario cambiarlo, pues solo se usa para
48
     * configuraciones internas de la biblioteca y no para las configuraciones
49
     * de "lo que usa" la biblioteca.
50
     *
51
     * @var string
52
     */
53
    protected string $prefix = 'derafu.lib.';
54
55
    /**
56
     * Define un mapa de parámetros (índice) y de qué configuración (valor)
57
     * obtener el valor que se asignará a dicho parámetro.
58
     *
59
     * @var array
60
     */
61
    protected array $parameters = [
62
        'kernel.project_dir' => 'app.project_dir',
63
    ];
64
65
    /**
66
     * Instancia que administra los datos de la configuración.
67
     *
68
     * @var DataContainerInterface
69
     */
70
    private DataContainerInterface $data;
71
72
    /**
73
     * Esquema de la estructura de los datos de la configuración.
74
     *
75
     * @var array
76
     */
77
    protected array $dataSchema = [
78
        'app' => [
79
            'types' => 'array',
80
            'default' => [],
81
            'schema' => [
82
                'project_dir' => [
83
                    'types' => 'string',
84
                ],
85
            ],
86
        ],
87
        'derafu' => [
88
            'types' => 'array',
89
            'default' => [],
90
            'schema' => [
91
                'lib' => [
92
                    'types' => 'array',
93
                    'default' => [],
94
                    'schema' => [
95
                        'services' => [
96
                            'types' => 'array',
97
                            'default' => [],
98
                            'schema' => [
99
                                'prefix' => [
100
                                    'types' => 'string',
101
                                    'default' => 'derafu.lib.',
102
                                ],
103
                                'kernelClass' => [
104
                                    'types' => 'string',
105
                                    'default' => Kernel::class,
106
                                ],
107
                                'serviceRegistryClass' => [
108
                                    'types' => 'string',
109
                                    'default' => ServiceRegistry::class,
110
                                ],
111
                                'compilerPassClass' => [
112
                                    'types' => 'string',
113
                                    'default' => ServiceProcessingCompilerPass::class,
114
                                ],
115
                            ],
116
                        ],
117
                    ],
118
                ],
119
                'packages' => [
120
                    'types' => 'array',
121
                    'default' => [],
122
                ],
123
            ],
124
        ],
125
    ];
126
127
    /**
128
     * Constructor de la configuración.
129
     *
130
     * @param string|array|null $config Archivo de configuración que se debe
131
     * cargar o el arreglo con la configuración.
132
     */
133 16
    public function __construct(string|array|null $config = null)
134
    {
135 16
        $this->load($config, $this->dataSchema);
136 16
        $this->configure();
137
    }
138
139
    /**
140
     * Carga la configuración de la aplicación desde un archivo.
141
     *
142
     * @param string $config
143
     * @return static
144
     */
145 16
    protected function load(string|array|null $config, array $schema): static
146
    {
147 16
        $config = $config ?? $this->resolvePath('config/config.yaml');
148
149 16
        $config = is_array($config)
150
            ? $config
151 16
            : (array) Yaml::parseFile($config)
152 16
        ;
153
154 16
        $this->data = new DataContainer($config, $schema);
155
156 16
        return $this;
157
    }
158
159
    /**
160
     * Realiza las configuraciones de la aplicación según lo que se haya
161
     * cargado como configuración.
162
     *
163
     * @return void
164
     */
165 16
    protected function configure(): void
166
    {
167 16
        if ($this->data->get('app.project_dir') === null) {
168 16
            $this->data->set('app.project_dir', $this->getProjectDir());
169
        }
170
    }
171
172
    /**
173
     * {@inheritdoc}
174
     */
175 16
    public function get(string $key, mixed $default = null): mixed
176
    {
177 16
        return $this->data->get($key, $default);
178
    }
179
180
    /**
181
     * {@inheritdoc}
182
     */
183 16
    public function getParameters(): array
184
    {
185 16
        $parameters = [];
186 16
        foreach ($this->parameters as $name => $source) {
187 16
            $value = $this->get($source);
188 16
            if ($value !== null) {
189 16
                $parameters[$name] = $value;
190
            }
191
        }
192
193 16
        return $parameters;
194
    }
195
196
    /**
197
     * {@inheritdoc}
198
     */
199 16
    public function getServicesPrefix(): string
200
    {
201 16
        return $this->get($this->prefix . 'services.prefix');
202
    }
203
204
    /**
205
     * {@inheritdoc}
206
     */
207 16
    public function getKernelClass(): string
208
    {
209 16
        return $this->get($this->prefix . 'services.kernelClass');
210
    }
211
212
    /**
213
     * {@inheritdoc}
214
     */
215 16
    public function getServiceRegistryClass(): string
216
    {
217 16
        return $this->get($this->prefix . 'services.serviceRegistryClass');
218
    }
219
220
    /**
221
     * {@inheritdoc}
222
     */
223 16
    public function getCompilerPassClass(): ?string
224
    {
225 16
        return $this->get($this->prefix . 'services.compilerPassClass');
226
    }
227
228
    /**
229
     * {@inheritdoc}
230
     */
231 16
    protected function getProjectDir(): string
232
    {
233 16
        if (!isset($this->projectDir)) {
234 16
            $vendorDir = InstalledVersions::getRootPackage();
235 16
            $this->projectDir = realpath($vendorDir['install_path']);
236
        }
237
238 16
        return $this->projectDir;
239
    }
240
241
    /**
242
     * {@inheritdoc}
243
     */
244 10
    public function resolvePath(?string $path = null): string
245
    {
246 10
        if (!$path) {
247
            return $this->getProjectDir();
248
        }
249
250 10
        if ($path[0] === '/') {
251
            return $path;
252
        }
253
254 10
        return $this->getProjectDir() . '/' . $path;
255
    }
256
257
    /**
258
     * {@inheritdoc}
259
     */
260 10
    public function getPackageConfiguration(
261
        string $package
262
    ): array|DataContainerInterface {
263 10
        $prefix = explode('.', $this->prefix)[0] . '.';
264 10
        $config = $this->data->get($prefix . 'packages.' . $package);
265
266 10
        return $config ?? [];
267
    }
268
}
269