Passed
Pull Request — main (#92)
by Andrey
12:24
created

HttpBuilder::parse()   A

Complexity

Conditions 4
Paths 8

Size

Total Lines 16
Code Lines 9

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 6
CRAP Score 4

Importance

Changes 1
Bugs 0 Features 0
Metric Value
cc 4
eloc 9
c 1
b 0
f 0
nc 8
nop 2
dl 0
loc 16
ccs 6
cts 6
cp 1
crap 4
rs 9.9666
1
<?php
2
3
namespace Helldar\Support\Helpers;
4
5
use ArgumentCountError;
6
use Helldar\Support\Facades\Helpers\Arr as ArrFacade;
7
use Helldar\Support\Facades\Helpers\Http as HttpHelper;
8
use Helldar\Support\Facades\Helpers\Instance as InstanceHelper;
9
use Helldar\Support\Facades\Helpers\Str as StrFacade;
10
use Helldar\Support\Tools\HttpBuilderPrepare;
11
use RuntimeException;
12
13
/**
14
 * Based on code by Maksim (Ellrion) Platonov.
15
 *
16
 * @see https://gist.github.com/Ellrion/f51ba0d40ae1d62eeae44fd1adf7b704
17
 *
18
 * @method HttpBuilder setFragment(array|string $value)
19
 * @method HttpBuilder setHost(string $value)
20
 * @method HttpBuilder setPass(string $value)
21
 * @method HttpBuilder setPath(string $value)
22
 * @method HttpBuilder setPort(string $value)
23
 * @method HttpBuilder setQuery(array|string $value)
24
 * @method HttpBuilder setScheme(string $value)
25
 * @method HttpBuilder setUser(string $value)
26
 * @method HttpBuilder putQuery(string $key, $value)
27
 * @method HttpBuilder removeQuery(string $key)
28
 * @method string|null getFragment()
29
 * @method string|null getHost()
30
 * @method string|null getPass()
31
 * @method string|null getPath()
32
 * @method string|null getPort()
33
 * @method string|null getQuery()
34
 * @method string|null getScheme()
35
 * @method string|null getUser()
36
 */
37
final class HttpBuilder
38
{
39
    protected $parsed = [];
40
41
    protected $components = [
42
        PHP_URL_SCHEME   => 'scheme',
43
        PHP_URL_HOST     => 'host',
44
        PHP_URL_PORT     => 'port',
45
        PHP_URL_USER     => 'user',
46
        PHP_URL_PASS     => 'pass',
47
        PHP_URL_QUERY    => 'query',
48
        PHP_URL_PATH     => 'path',
49
        PHP_URL_FRAGMENT => 'fragment',
50
    ];
51
52
    protected $allow_put_remove = ['query'];
53
54
    protected $casts = [
55
        'query' => 'array',
56
    ];
57
58 56
    /**
59
     * Calling magic methods.
60 56
     *
61 56
     * @param  string  $method
62
     * @param  mixed  $args
63 56
     *
64 4
     * @return $this|string|null
65
     */
66
    public function __call(string $method, $args)
67
    {
68 52
        if ($this->isGetter($method) || $this->isSetter($method) || $this->isPutter($method) || $this->isRemover($method)) {
69 48
            $key = $this->parseKey($method);
70
71 48
            if (! $this->allowKey($key) || ! $this->allowArrayable($method, $key)) {
72
                throw new RuntimeException($method . ' method not defined.');
73 24
            }
74 24
75
            switch (true) {
76 20
                case $this->isGetter($method):
77
                    $this->validateArgumentsCount($method, $args, 0);
78
79
                    return $this->get($key);
0 ignored issues
show
Bug introduced by
It seems like $key can also be of type null; however, parameter $key of Helldar\Support\Helpers\HttpBuilder::get() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

79
                    return $this->get(/** @scrutinizer ignore-type */ $key);
Loading history...
Bug Best Practice introduced by
The expression return $this->get($key) also could return the type array which is incompatible with the documented return type Helldar\Support\Helpers\HttpBuilder|null|string.
Loading history...
80
81
                case $this->isSetter($method):
82
                    $this->validateArgumentsCount($method, $args);
83
84
                    return $this->set($key, $args[0]);
0 ignored issues
show
Bug introduced by
It seems like $key can also be of type null; however, parameter $key of Helldar\Support\Helpers\HttpBuilder::set() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

84
                    return $this->set(/** @scrutinizer ignore-type */ $key, $args[0]);
Loading history...
85
86
                case $this->isPutter($method):
87
                    $this->validateArgumentsCount($method, $args, 2);
88 8
89
                    return $this->put($key, $args[0], $args[1]);
0 ignored issues
show
Bug introduced by
It seems like $key can also be of type null; however, parameter $key of Helldar\Support\Helpers\HttpBuilder::put() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

89
                    return $this->put(/** @scrutinizer ignore-type */ $key, $args[0], $args[1]);
Loading history...
90 8
91
                case $this->isRemover($method):
92
                    $this->validateArgumentsCount($method, $args);
93
94
                    return $this->remove($key, $args[0]);
0 ignored issues
show
Bug introduced by
It seems like $key can also be of type null; however, parameter $key of Helldar\Support\Helpers\HttpBuilder::remove() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

94
                    return $this->remove(/** @scrutinizer ignore-type */ $key, $args[0]);
Loading history...
95
            }
96
        }
97
98
        throw new RuntimeException("Using an unknown method: \"{$method}\"");
99
    }
100
101 32
    /**
102
     * Gets the current instance of the object.
103 32
     *
104 28
     * @return $this
105
     */
106
    public function same(): self
107 26
    {
108 26
        return $this;
109
    }
110 26
111 22
    /**
112 4
     * Parse a URL.
113
     *
114 26
     * @param  string|null  $url
115
     * @param  int  $component
116
     *
117
     * @return $this
118
     */
119
    public function parse(?string $url, int $component = -1): self
120
    {
121
        if ($component === -1) {
122
            HttpHelper::validateUrl($url);
123
        }
124 6
125
        $component = $this->componentIndex($component);
126 6
        $key       = $this->componentKey($component);
127 6
128 2
        $component === -1 || empty($key)
129
            ? $this->parsed       = parse_url($url)
130
            : $this->parsed[$key] = parse_url($url, $component);
131 4
132
        $this->cast();
133
134 4
        return $this;
135
    }
136
137
    /**
138
     * Filling the builder with parsed data.
139
     *
140
     * @param  array  $parsed
141
     *
142 12
     * @return $this
143
     */
144 12
    public function raw(array $parsed): self
145 12
    {
146 12
        foreach ($parsed as $key => $value) {
147
            if (! $this->allowKey($key)) {
148
                throw new RuntimeException('Filling in the "' . $key . '" key is prohibited.');
149
            }
150
151
            $this->set($key, $value);
152
        }
153
154 4
        $this->cast();
155
156
        return $this;
157 4
    }
158 4
159 4
    /**
160 4
     * Compiles parameters to URL.
161 4
     *
162 4
     * @return string
163 4
     */
164 4
    public function compile(): string
165
    {
166
        return implode('', array_filter(array_map(function ($value) {
167
            return InstanceHelper::of($value, HttpBuilderPrepare::class) ? $value->get() : $value;
168
        }, $this->prepare())));
169
    }
170
171
    /**
172
     * Returns parsed data.
173 12
     *
174
     * @return null[]|string[]
175
     */
176 12
    public function toArray(): array
177 12
    {
178 12
        return [
179
            'scheme'   => $this->getScheme(),
180 12
            'host'     => $this->getHost(),
181
            'port'     => $this->getPort(),
182 12
            'user'     => $this->getUser(),
183 12
            'pass'     => $this->getPass(),
184 12
            'query'    => $this->getQuery(),
185 12
            'path'     => $this->getPath(),
186 12
            'fragment' => $this->getFragment(),
187
        ];
188
    }
189
190
    /**
191
     * Prepares data for compilation.
192
     *
193
     * @return array
194
     */
195
    protected function prepare(): array
196
    {
197 26
        return [
198
            HttpBuilderPrepare::make()->of($this->getScheme())->suffix('://'),
199 26
            HttpBuilderPrepare::make()->of($this->getUser()),
200
            HttpBuilderPrepare::make()->of($this->getPass())->prefix(':'),
201
202
            $this->getUser() || $this->getPass() ? '@' : '',
203
204
            HttpBuilderPrepare::make()->of($this->getHost()),
205
            HttpBuilderPrepare::make()->of($this->getPort())->prefix(':'),
206
            HttpBuilderPrepare::make()->of($this->getPath())->prefix('/'),
207
            HttpBuilderPrepare::make()->of($this->getQuery())->prefix('?'),
208
            HttpBuilderPrepare::make()->of($this->getFragment())->prefix('#'),
209 26
        ];
210
    }
211 26
212
    /**
213
     * Gets the index of the component.
214
     *
215
     * @param  int  $component
216
     *
217
     * @return int
218
     */
219
    protected function componentIndex(int $component = -1): int
220
    {
221 58
        return ArrFacade::getKey($this->components, $component, -1);
222
    }
223 58
224
    /**
225
     * Gets the key for the component.
226
     *
227
     * @param  int  $component
228
     *
229
     * @return string|null
230
     */
231
    protected function componentKey(int $component = -1): ?string
232
    {
233 56
        return ArrFacade::get($this->components, $component);
234
    }
235 56
236
    /**
237
     * Checks if calling the requested key is allowed.
238
     *
239
     * @param  string|null  $key
240
     *
241
     * @return bool
242
     */
243
    protected function allowKey(?string $key): bool
244
    {
245 26
        return in_array($key, $this->components);
246
    }
247 26
248
    protected function allowArrayable(?string $method, ?string $key): bool
249
    {
250
        if ($this->isPutter($method) || $this->isRemover($method)) {
0 ignored issues
show
Bug introduced by
It seems like $method can also be of type null; however, parameter $method of Helldar\Support\Helpers\HttpBuilder::isRemover() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

250
        if ($this->isPutter($method) || $this->isRemover(/** @scrutinizer ignore-type */ $method)) {
Loading history...
Bug introduced by
It seems like $method can also be of type null; however, parameter $method of Helldar\Support\Helpers\HttpBuilder::isPutter() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

250
        if ($this->isPutter(/** @scrutinizer ignore-type */ $method) || $this->isRemover($method)) {
Loading history...
251
            return in_array($key, $this->allow_put_remove);
252
        }
253
254
        return true;
255
    }
256
257 56
    /**
258
     * Checks if the method is a request for information.
259 56
     *
260
     * @param  string  $method
261 56
     *
262
     * @return bool
263
     */
264
    protected function isGetter(string $method): bool
265
    {
266
        return StrFacade::startsWith($method, 'get');
267
    }
268
269
    /**
270
     * Checks if the method is a request to fill information.
271
     *
272 24
     * @param  string  $method
273
     *
274 24
     * @return bool
275
     */
276 24
    protected function isSetter(string $method): bool
277
    {
278
        return StrFacade::startsWith($method, 'set');
279
    }
280
281
    /**
282
     * Checks if the method is a request to put value.
283
     *
284
     * @param  string  $method
285
     *
286 48
     * @return bool
287
     */
288 48
    protected function isPutter(string $method): bool
289
    {
290
        return StrFacade::startsWith($method, 'put');
291
    }
292
293
    /**
294
     * Checks if the method is a request to remove value.
295
     *
296
     * @param  string  $method
297
     *
298 52
     * @return bool
299
     */
300 52
    protected function isRemover(string $method): bool
301 4
    {
302
        return StrFacade::startsWith($method, 'remove');
303 48
    }
304
305
    /**
306
     * Gets the key of the component from the name of the magic method.
307
     *
308
     * @param  string  $method
309
     *
310
     * @return string|null
311
     */
312
    protected function parseKey(string $method): ?string
313
    {
314
        $search = 'unknown';
315
316
        switch (true) {
317
            case $this->isGetter($method):
318
                $search = 'get';
319
                break;
320
321
            case $this->isSetter($method):
322
                $search = 'set';
323
                break;
324
325
            case $this->isPutter($method):
326
                $search = 'put';
327
                break;
328
329
            case $this->isRemover($method):
330
                $search = 'remove';
331
                break;
332
        }
333
334
        return StrFacade::lower(StrFacade::after($method, $search));
335
    }
336
337
    /**
338
     * Set the component key with a value.
339
     *
340
     * @param  string  $key
341
     * @param  mixed  $value
342
     *
343
     * @return $this
344
     */
345
    protected function set(string $key, $value): self
346
    {
347
        switch (true) {
348
            case $this->hasCastArray($key):
349
                $value = $this->castToArray($value);
350
                break;
351
        }
352
353
        $this->parsed[$key] = $value;
354
355
        return $this;
356
    }
357
358
    /**
359
     * Gets the value of the component.
360
     *
361
     * @param  string  $key
362
     *
363
     * @return array|string|null
364
     */
365
    protected function get(string $key)
366
    {
367
        if ($value = $this->parsed[$key] ?? null) {
368
            return $this->hasCastArray($key) ? http_build_query($value) : $value;
369
        }
370
371
        return null;
372
    }
373
374
    /**
375
     * Adds a key-value to an array.
376
     *
377
     * @param  string  $key
378
     * @param  string  $parameter
379
     * @param  mixed  $value
380
     *
381
     * @return $this
382
     */
383
    protected function put(string $key, string $parameter, $value): self
384
    {
385
        $this->parsed[$key][$parameter] = $value;
386
387
        return $this;
388
    }
389
390
    /**
391
     * Removes a key from a variable.
392
     *
393
     * @param  string  $key
394
     * @param  string  $parameter
395
     *
396
     * @return $this
397
     */
398
    protected function remove(string $key, string $parameter): self
399
    {
400
        unset($this->parsed[$key][$parameter]);
401
402
        return $this;
403
    }
404
405
    protected function hasCastArray(string $key): bool
406
    {
407
        return ($this->casts[$key] ?? null) === 'array';
408
    }
409
410
    protected function cast(): void
411
    {
412
        foreach ($this->casts as $key => $cast) {
413
            $value = $this->parsed[$key] ?? null;
414
415
            switch ($cast) {
416
                case 'array':
417
                    $value = $this->castToArray($value);
418
                    break;
419
            }
420
421
            $this->parsed[$key] = $value;
422
        }
423
    }
424
425
    protected function castToArray($value): array
426
    {
427
        if (empty($value)) {
428
            return [];
429
        }
430
431
        if (is_array($value)) {
432
            return $value;
433
        }
434
435
        $items = [];
436
437
        foreach (explode('&', $value) as $item) {
438
            [$key, $value] = StrFacade::contains($item, '=') ? explode('=', $item) : [0, $item];
439
440
            $items[$key] = $value;
441
        }
442
443
        return $items;
444
    }
445
446
    /**
447
     * Checks the number of arguments passed.
448
     *
449
     * @param  string  $method
450
     * @param  array  $args
451
     * @param  int  $need
452
     */
453
    protected function validateArgumentsCount(string $method, array $args, int $need = 1): void
454
    {
455
        if (count($args) > $need) {
456
            throw new ArgumentCountError($method . ' expects at most ' . $need . ' parameter, ' . count($args) . ' given.');
457
        }
458
    }
459
}
460