Passed
Pull Request — master (#42)
by Dmitriy
15:39
created

Builder::path()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 3
Code Lines 1

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 1
eloc 1
nc 1
nop 2
dl 0
loc 3
rs 10
c 0
b 0
f 0
1
<?php
2
3
namespace Yiisoft\Composer\Config;
4
5
use JsonException;
6
use Yiisoft\Composer\Config\Config\Config;
7
use Yiisoft\Composer\Config\Config\ConfigFactory;
8
use Yiisoft\Composer\Config\Util\Resolver;
9
10
use function dirname;
11
12
/**
13
 * Builder assembles config files.
14
 */
15
class Builder
16
{
17
    private const OUTPUT_DIR_SUFFIX = '-output';
18
19
    /**
20
     * @var string path to the Composer project root
21
     */
22
    private string $baseDir;
23
24
    /**
25
     * @var string path to output assembled configs
26
     */
27
    private string $outputDir;
28
29
    /**
30
     * @var Config[] configurations
31
     */
32
    private array $configs = [];
33
34
    private ConfigFactory $configFactory;
35
36
    /**
37
     * Builder constructor.
38
     *
39
     * @param ConfigFactory $configFactory
40
     * @param string $baseDir path to the Composer project root
41
     */
42
    public function __construct(ConfigFactory $configFactory, string $baseDir)
43
    {
44
        $this->configFactory = $configFactory;
45
        $this->baseDir = $baseDir;
46
        $this->outputDir = self::findOutputDir($baseDir);
47
    }
48
49
    public function createAlternative($name): Builder
50
    {
51
        $alt = new static($this->configFactory, $this->baseDir);
52
        $alt->setOutputDir($this->outputDir . DIRECTORY_SEPARATOR . $name);
53
        foreach (['aliases', 'packages'] as $key) {
54
            $alt->configs[$key] = $this->getConfig($key)->clone($alt);
55
        }
56
57
        return $alt;
58
    }
59
60
    public function setOutputDir(?string $outputDir): void
61
    {
62
        $this->outputDir = $outputDir
63
            ? static::buildAbsPath($this->getBaseDir(), $outputDir)
64
            : static::findOutputDir($this->getBaseDir());
65
    }
66
67
    public static function rebuild(string $baseDir = null): void
68
    {
69
        if ($baseDir === null) {
70
            $baseDir = static::findDirContainsComposerJsonRecursively(getcwd());
71
        }
72
        $builder = new self(new ConfigFactory(), $baseDir);
73
        $files = $builder->getConfig('__files')->load();
74
75
        $builder->buildUserConfigs($files->getValues());
76
    }
77
78
    /**
79
     * Returns default output dir.
80
     *
81
     * @param string|null $baseDir path to the root Composer package. When `null`,
82
     * @return string
83
     * @throws JsonException
84
     */
85
    private static function findOutputDir(string $baseDir = null): string
86
    {
87
        if ($baseDir === null) {
88
            $baseDir = static::findDirContainsComposerJsonRecursively(getcwd());
89
        }
90
        $path = $baseDir . DIRECTORY_SEPARATOR . 'composer.json';
91
        $data = @json_decode(file_get_contents($path), true);
92
        $dir = $data['extra'][Package::EXTRA_OUTPUT_DIR_OPTION_NAME] ?? null;
93
94
        return $dir ? static::buildAbsPath($baseDir, $dir) : static::defaultOutputDir($baseDir);
95
    }
96
97
    private static function findDirContainsComposerJsonRecursively(string $cwd): string
98
    {
99
        if (file_exists($cwd . DIRECTORY_SEPARATOR . 'composer.json')) {
100
            return $cwd;
101
        }
102
103
        return static::findDirContainsComposerJsonRecursively(dirname($cwd));
104
    }
105
106
    /**
107
     * Returns default output dir.
108
     *
109
     * @param string $baseDir path to base directory
110
     * @return string
111
     */
112
    private static function defaultOutputDir(string $baseDir = null): string
113
    {
114
        if ($baseDir) {
115
            $dir = $baseDir . DIRECTORY_SEPARATOR . 'vendor' . DIRECTORY_SEPARATOR . 'yiisoft' . DIRECTORY_SEPARATOR . basename(dirname(__DIR__));
116
        } else {
117
            $dir = dirname(__DIR__);
118
        }
119
120
        return $dir . static::OUTPUT_DIR_SUFFIX;
121
    }
122
123
    /**
124
     * Returns full path to assembled config file.
125
     *
126
     * @param string $filename name of config
127
     * @param string $baseDir path to base dir
128
     * @return string absolute path
129
     * @throws JsonException
130
     */
131
    public static function path(string $filename, string $baseDir = null): string
132
    {
133
        return static::buildAbsPath(static::findOutputDir($baseDir), $filename . '.php');
134
    }
135
136
    private static function buildAbsPath(string $dir, string $file): string
137
    {
138
        return self::isAbsolutePath($file) ? $file : $dir . DIRECTORY_SEPARATOR . $file;
139
    }
140
141
    private static function isAbsolutePath(string $path): string
142
    {
143
        return strpos($path, '/') === 0 || strpos($path, ':') === 1 || strpos($path, '\\\\') === 0;
0 ignored issues
show
Bug Best Practice introduced by
The expression return strpos($path, '/'...rpos($path, '\\') === 0 returns the type boolean which is incompatible with the type-hinted return string.
Loading history...
144
    }
145
146
    /**
147
     * Builds all (user and system) configs by given files list.
148
     *
149
     * @param null|array $files files to process: config name => list of files
150
     */
151
    public function buildAllConfigs(array $files): void
152
    {
153
        $this->buildUserConfigs($files);
154
        $this->buildSystemConfigs($files);
155
    }
156
157
    /**
158
     * Builds configs by given files list.
159
     *
160
     * @param null|array $files files to process: config name => list of files
161
     * @return array
162
     */
163
    private function buildUserConfigs(array $files): array
164
    {
165
        $resolver = new Resolver($files);
166
        $files = $resolver->get();
167
        foreach ($files as $name => $paths) {
168
            $this->getConfig($name)->load($paths)->build()->write();
169
        }
170
171
        return $files;
172
    }
173
174
    private function buildSystemConfigs(array $files): void
175
    {
176
        $this->getConfig('__files')->setValues($files);
177
        foreach (['__files', 'aliases', 'packages'] as $name) {
178
            $this->getConfig($name)->build()->write();
179
        }
180
    }
181
182
    public function getOutputPath(string $name): string
183
    {
184
        return $this->outputDir . DIRECTORY_SEPARATOR . $name . '.php';
185
    }
186
187
    public function getConfig(string $name)
188
    {
189
        if (!array_key_exists($name, $this->configs)) {
190
            $this->configs[$name] = $this->configFactory->create($this, $name);
191
        }
192
193
        return $this->configs[$name];
194
    }
195
196
    public function getVars(): array
197
    {
198
        $vars = [];
199
        foreach ($this->configs as $name => $config) {
200
            $vars[$name] = $config->getValues();
201
        }
202
203
        return $vars;
204
    }
205
206
    public function mergeAliases(array $aliases): void
207
    {
208
        $this->getConfig('aliases')->mergeValues($aliases);
209
    }
210
211
    public function setPackage(string $name, array $data): void
212
    {
213
        $this->getConfig('packages')->setValue($name, $data);
214
    }
215
216
    /**
217
     * @return string a full path to the project root
218
     */
219
    public function getBaseDir(): string
220
    {
221
        return $this->baseDir;
222
    }
223
}
224