Completed
Push — master ( 78d4e7...979ecb )
by smiley
02:57
created

Uri   F

Complexity

Total Complexity 89

Size/Duplication

Total Lines 685
Duplicated Lines 0 %

Coupling/Cohesion

Components 1
Dependencies 0

Importance

Changes 0
Metric Value
dl 0
loc 685
rs 1.915
c 0
b 0
f 0
wmc 89
lcom 1
cbo 0

36 Methods

Rating   Name   Duplication   Size   Complexity  
A __construct() 0 13 3
B __toString() 0 26 6
A filterScheme() 0 8 2
A getScheme() 0 3 1
A withScheme() 0 15 2
A filterUser() 0 8 2
A filterPass() 0 8 2
A getAuthority() 0 14 3
A getUserInfo() 0 3 2
A withUserInfo() 0 19 3
A filterHost() 0 12 3
A getHost() 0 3 1
A withHost() 0 14 2
A filterPort() 0 14 4
A getPort() 0 3 1
A withPort() 0 15 2
A filterPath() 0 8 2
A getPath() 0 3 1
A withPath() 0 14 2
A filterQuery() 0 8 2
A getQuery() 0 3 1
A withQuery() 0 14 2
A filterFragment() 0 3 1
A getFragment() 0 3 1
A withFragment() 0 14 2
A parseUriParts() 0 13 3
A replaceChars() 0 14 2
A removeDefaultPort() 0 7 4
B validateState() 0 26 10
A fromParts() 0 8 1
A isAbsolute() 0 3 1
A isNetworkPathReference() 0 3 2
A isAbsolutePathReference() 0 3 4
A isRelativePathReference() 0 3 4
A withoutQueryValue() 0 16 2
A withQueryValue() 0 26 3

How to fix   Complexity   

Complex Class

Complex classes like Uri often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes. You can also have a look at the cohesion graph to spot any un-connected, or weakly-connected components.

Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.

While breaking up the class, it is a good idea to analyze how other classes use Uri, and based on these observations, apply Extract Interface, too.

1
<?php
2
/**
3
 * Class Uri
4
 *
5
 * @filesource   Uri.php
6
 * @created      10.08.2018
7
 * @package      chillerlan\HTTP\Psr7
8
 * @author       smiley <[email protected]>
9
 * @copyright    2018 smiley
10
 * @license      MIT
11
 */
12
13
namespace chillerlan\HTTP\Psr7;
14
15
use InvalidArgumentException;
16
use Psr\Http\Message\UriInterface;
17
18
final class Uri implements UriInterface{
19
20
	private const DEFAULT_PORTS = [
21
		'http'   => 80,
22
		'https'  => 443,
23
		'ftp'    => 21,
24
		'gopher' => 70,
25
		'nntp'   => 119,
26
		'news'   => 119,
27
		'telnet' => 23,
28
		'tn3270' => 23,
29
		'imap'   => 143,
30
		'pop'    => 110,
31
		'ldap'   => 389,
32
	];
33
34
	/**
35
	 * @var string
36
	 */
37
	private $scheme = '';
38
39
	/**
40
	 * @var string
41
	 */
42
	private $user = '';
43
44
	/**
45
	 * @var string
46
	 */
47
	private $pass = '';
48
49
	/**
50
	 * @var string
51
	 */
52
	private $host = '';
53
54
	/**
55
	 * @var int
56
	 */
57
	private $port = null;
58
59
	/**
60
	 * @var string
61
	 */
62
	private $path = '';
63
64
	/**
65
	 * @var string
66
	 */
67
	private $query = '';
68
69
	/**
70
	 * @var string
71
	 */
72
	private $fragment = '';
73
74
	/**
75
	 * Uri constructor.
76
	 *
77
	 * @param string|null $uri
78
	 *
79
	 * @throws \InvalidArgumentException
80
	 */
81
	public function __construct(string $uri = null){
82
83
		if($uri !== ''){
84
			$parts = parse_url($uri);
85
86
			if($parts === false){
87
				throw new InvalidArgumentException('invalid URI: "'.$uri.'"');
88
			}
89
90
			$this->parseUriParts($parts);
91
		}
92
93
	}
94
95
	/**
96
	 * @inheritdoc
97
	 */
98
	public function __toString(){
99
		$this->validateState();
100
101
		$uri       = '';
102
		$authority = $this->getAuthority();
103
104
		if($this->scheme !== ''){
105
			$uri .= $this->scheme.':';
106
		}
107
108
		if($authority !== '' || $this->scheme === 'file'){
109
			$uri .= '//'.$authority;
110
		}
111
112
		$uri .= $this->path;
113
114
		if($this->query !== ''){
115
			$uri .= '?'.$this->query;
116
		}
117
118
		if($this->fragment !== ''){
119
			$uri .= '#'.$this->fragment;
120
		}
121
122
		return $uri;
123
	}
124
125
	/**
126
	 * Scheme
127
	 */
128
129
	/**
130
	 * @param mixed $scheme
131
	 *
132
	 * @return string
133
	 * @throws \InvalidArgumentException
134
	 */
135
	private function filterScheme($scheme):string{
136
137
		if(!is_string($scheme)){
138
			throw new InvalidArgumentException(sprintf('scheme must be a string'));
139
		}
140
141
		return strtolower($scheme);
142
	}
143
144
	/**
145
	 * @inheritdoc
146
	 */
147
	public function getScheme():string{
148
		return $this->scheme;
149
	}
150
151
	/**
152
	 * @inheritdoc
153
	 */
154
	public function withScheme($scheme):UriInterface{
155
		$scheme = $this->filterScheme($scheme);
156
157
		if($this->scheme === $scheme){
158
			return $this;
159
		}
160
161
		$clone         = clone $this;
162
		$clone->scheme = $scheme;
163
164
		$clone->removeDefaultPort();
165
		$clone->validateState();
166
167
		return $clone;
168
	}
169
170
	/**
171
	 * Authority
172
	 */
173
174
	/**
175
	 * @param mixed $user
176
	 *
177
	 * @return string
178
	 * @throws \InvalidArgumentException
179
	 */
180
	private function filterUser($user):string{
181
182
		if(!is_string($user)){
183
			throw new InvalidArgumentException(sprintf('user must be a string'));
184
		}
185
186
		return $user;
187
	}
188
189
	/**
190
	 * @param mixed $pass
191
	 *
192
	 * @return string
193
	 * @throws \InvalidArgumentException
194
	 */
195
	private function filterPass($pass):string{
196
197
		if(!is_string($pass)){
198
			throw new InvalidArgumentException(sprintf('pass must be a string'));
199
		}
200
201
		return $pass;
202
	}
203
204
	/**
205
	 * @inheritdoc
206
	 */
207
	public function getAuthority():string{
208
		$authority = $this->host;
209
		$userInfo  = $this->getUserInfo();
210
211
		if($userInfo !== ''){
212
			$authority = $userInfo.'@'.$authority;
213
		}
214
215
		if($this->port !== null){
216
			$authority .= ':'.$this->port;
217
		}
218
219
		return $authority;
220
	}
221
222
	/**
223
	 * @inheritdoc
224
	 */
225
	public function getUserInfo():string{
226
		return (string)$this->user.($this->pass != '' ? ':'.$this->pass : '');
227
	}
228
229
	/**
230
	 * @inheritdoc
231
	 */
232
	public function withUserInfo($user, $password = null):UriInterface{
233
		$info = $user;
234
235
		if($password !== ''){
236
			$info .= ':'.$password;
237
		}
238
239
		if($this->getUserInfo() === $info){
240
			return $this;
241
		}
242
243
		$clone       = clone $this;
244
		$clone->user = $user;
245
		$clone->pass = $password;
246
247
		$clone->validateState();
248
249
		return $clone;
250
	}
251
252
	/**
253
	 * Host
254
	 */
255
256
	/**
257
	 * @param mixed $host
258
	 *
259
	 * @return string
260
	 * @throws \InvalidArgumentException
261
	 */
262
	private function filterHost($host):string{
263
264
		if(!is_string($host)){
265
			throw new InvalidArgumentException('host must be a string');
266
		}
267
268
		if(filter_var($host, FILTER_VALIDATE_IP, FILTER_FLAG_IPV6)){
269
			$host = '['.$host.']';
270
		}
271
272
		return strtolower($host);
273
	}
274
275
	/**
276
	 * @inheritdoc
277
	 */
278
	public function getHost():string{
279
		return $this->host;
280
	}
281
282
	/**
283
	 * @inheritdoc
284
	 */
285
	public function withHost($host):UriInterface{
286
		$host = $this->filterHost($host);
287
288
		if($this->host === $host){
289
			return $this;
290
		}
291
292
		$clone       = clone $this;
293
		$clone->host = $host;
294
295
		$clone->validateState();
296
297
		return $clone;
298
	}
299
300
	/**
301
	 * Port
302
	 */
303
304
	/**
305
	 * @param mixed $port
306
	 *
307
	 * @return int|null
308
	 * @throws \InvalidArgumentException
309
	 */
310
	private function filterPort($port):?int{
311
312
		if($port === null){
313
			return null;
314
		}
315
316
		$port = (int)$port;
317
318
		if($port >= 1 && $port <= 0xffff){
319
			return $port;
320
		}
321
322
		throw new InvalidArgumentException(sprintf('invalid port: %d', $port));
323
	}
324
325
	/**
326
	 * @inheritdoc
327
	 */
328
	public function getPort():?int{
329
		return $this->port;
330
	}
331
332
	/**
333
	 * @inheritdoc
334
	 */
335
	public function withPort($port):UriInterface{
336
		$port = $this->filterPort($port);
337
338
		if($this->port === $port){
339
			return $this;
340
		}
341
342
		$clone       = clone $this;
343
		$clone->port = $port;
344
345
		$clone->removeDefaultPort();
346
		$clone->validateState();
347
348
		return $clone;
349
	}
350
351
	/**
352
	 * Path
353
	 */
354
355
	/**
356
	 * @param mixed $path
357
	 *
358
	 * @return string
359
	 * @throws \InvalidArgumentException
360
	 */
361
	private function filterPath($path):string{
362
363
		if(!is_string($path)){
364
			throw new InvalidArgumentException('path must be a string');
365
		}
366
367
		return $this->replaceChars($path);
368
	}
369
370
	/**
371
	 * @inheritdoc
372
	 */
373
	public function getPath():string{
374
		return $this->path;
375
	}
376
377
	/**
378
	 * @inheritdoc
379
	 */
380
	public function withPath($path):UriInterface{
381
		$path = $this->filterPath($path);
382
383
		if($this->path === $path){
384
			return $this;
385
		}
386
387
		$clone       = clone $this;
388
		$clone->path = $path;
389
390
		$clone->validateState();
391
392
		return $clone;
393
	}
394
395
	/**
396
	 * Query
397
	 */
398
399
	/**
400
	 * @param $query
401
	 *
402
	 * @return string
403
	 * @throws \InvalidArgumentException
404
	 */
405
	private function filterQuery($query):string{
406
407
		if(!is_string($query)){
408
			throw new InvalidArgumentException('query and fragment must be a string');
409
		}
410
411
		return $this->replaceChars($query, true);
412
	}
413
414
	/**
415
	 * @inheritdoc
416
	 */
417
	public function getQuery():string{
418
		return $this->query;
419
	}
420
421
	/**
422
	 * @inheritdoc
423
	 */
424
	public function withQuery($query):UriInterface{
425
		$query = $this->filterQuery($query);
426
427
		if($this->query === $query){
428
			return $this;
429
		}
430
431
		$clone        = clone $this;
432
		$clone->query = $query;
433
434
		$clone->validateState();
435
436
		return $clone;
437
	}
438
439
	/**
440
	 * Fragment
441
	 */
442
443
	/**
444
	 * @param $fragment
445
	 *
446
	 * @return string
447
	 */
448
	private function filterFragment($fragment):string{
449
		return $this->filterQuery($fragment);
450
	}
451
452
	/**
453
	 * @inheritdoc
454
	 */
455
	public function getFragment():string{
456
		return $this->fragment;
457
	}
458
459
	/**
460
	 * @inheritdoc
461
	 */
462
	public function withFragment($fragment):UriInterface{
463
		$fragment = $this->filterFragment($fragment);
464
465
		if($this->fragment === $fragment){
466
			return $this;
467
		}
468
469
		$clone           = clone $this;
470
		$clone->fragment = $fragment;
471
472
		$clone->validateState();
473
474
		return $clone;
475
	}
476
477
	/**
478
	 * @param array $parts
479
	 *
480
	 * @return void
481
	 */
482
	private function parseUriParts(array $parts):void{
483
484
		foreach(['scheme', 'user', 'pass', 'host', 'port', 'path', 'query', 'fragment'] as $part){
485
486
			if(!isset($parts[$part])){
487
				continue;
488
			}
489
490
			$this->{$part} = call_user_func_array([$this, 'filter'.ucfirst($part)], [$parts[$part]]);
491
		}
492
493
		$this->removeDefaultPort();
494
	}
495
496
	/**
497
	 * @param string    $str
498
	 * @param bool|null $query
499
	 *
500
	 * @return string
501
	 */
502
	private function replaceChars(string $str, bool $query = null):string{
503
		return preg_replace_callback(
504
			'/(?:[^'
505
			.'a-z\d_\-\.~'
506
			.'!\$&\'\(\)\*\+,;='
507
			.'%:@\/'.($query ? '\?' : '')
508
			.']++|%(?![a-f\d]{2}))/i',
509
			function(array $match):string{
510
				return rawurlencode($match[0]);
511
			},
512
			$str
513
		);
514
515
	}
516
517
	/**
518
	 * @return void
519
	 */
520
	private function removeDefaultPort():void{
521
522
		if($this->port !== null && (isset($this::DEFAULT_PORTS[$this->scheme]) && $this->port === $this::DEFAULT_PORTS[$this->scheme])){
523
			$this->port = null;
524
		}
525
526
	}
527
528
	/**
529
	 * @return void
530
	 */
531
	private function validateState():void{
532
533
		if(empty($this->host) && ($this->scheme === 'http' || $this->scheme === 'https')){
534
			$this->host = 'localhost';
535
		}
536
537
		if($this->getAuthority() !== ''){
538
539
			if(isset($this->path[0]) && $this->path[0] !== '/'){
540
				$this->path = '/'.$this->path; // automagically fix the path, unlike Guzzle
541
			}
542
543
		}
544
		else{
545
546
			if(strpos($this->path, '//') === 0){
547
				$this->path = '/'.ltrim($this->path, '/'); // automagically fix the path, unlike Guzzle
548
			}
549
550
			if(empty($this->scheme) && strpos(explode('/', $this->path, 2)[0], ':') !== false){
551
				throw new InvalidArgumentException('A relative URI must not have a path beginning with a segment containing a colon');
552
			}
553
554
		}
555
556
	}
557
558
	/**
559
	 * Additional methods
560
	 */
561
562
	/**
563
	 * @see \parse_url()
564
	 *
565
	 * @param array $parts
566
	 *
567
	 * @return \Psr\Http\Message\UriInterface|\chillerlan\HTTP\Psr7\Uri
568
	 */
569
	public static function fromParts(array $parts):UriInterface{
570
		$uri = new self;
571
572
		$uri->parseUriParts($parts);
573
		$uri->validateState();
574
575
		return $uri;
576
	}
577
578
	/**
579
	 * Whether the URI is absolute, i.e. it has a scheme.
580
	 *
581
	 * An instance of UriInterface can either be an absolute URI or a relative reference. This method returns true
582
	 * if it is the former. An absolute URI has a scheme. A relative reference is used to express a URI relative
583
	 * to another URI, the base URI. Relative references can be divided into several forms:
584
	 * - network-path references, e.g. '//example.com/path'
585
	 * - absolute-path references, e.g. '/path'
586
	 * - relative-path references, e.g. 'subpath'
587
	 *
588
	 * @return bool
589
	 * @see  Uri::isNetworkPathReference
590
	 * @see  Uri::isAbsolutePathReference
591
	 * @see  Uri::isRelativePathReference
592
	 * @link https://tools.ietf.org/html/rfc3986#section-4
593
	 */
594
	public function isAbsolute():bool{
595
		return $this->getScheme() !== '';
596
	}
597
598
	/**
599
	 * Whether the URI is a network-path reference.
600
	 *
601
	 * A relative reference that begins with two slash characters is termed an network-path reference.
602
	 *
603
	 * @return bool
604
	 * @link https://tools.ietf.org/html/rfc3986#section-4.2
605
	 */
606
	public function isNetworkPathReference():bool{
607
		return $this->getScheme() === '' && $this->getAuthority() !== '';
608
	}
609
610
	/**
611
	 * Whether the URI is a absolute-path reference.
612
	 *
613
	 * A relative reference that begins with a single slash character is termed an absolute-path reference.
614
	 *
615
	 * @return bool
616
	 * @link https://tools.ietf.org/html/rfc3986#section-4.2
617
	 */
618
	public function isAbsolutePathReference():bool{
619
		return $this->getScheme() === '' && $this->getAuthority() === '' && isset($this->getPath()[0]) && $this->getPath()[0] === '/';
620
	}
621
622
	/**
623
	 * Whether the URI is a relative-path reference.
624
	 *
625
	 * A relative reference that does not begin with a slash character is termed a relative-path reference.
626
	 *
627
	 * @return bool
628
	 * @link https://tools.ietf.org/html/rfc3986#section-4.2
629
	 */
630
	public function isRelativePathReference():bool{
631
		return $this->getScheme() === '' && $this->getAuthority() === '' && (!isset($this->getPath()[0]) || $this->getPath()[0] !== '/');
632
	}
633
634
	/**
635
	 * removes a specific query string value.
636
	 *
637
	 * Any existing query string values that exactly match the provided key are
638
	 * removed.
639
	 *
640
	 * @param string $key Query string key to remove.
641
	 *
642
	 * @return \Psr\Http\Message\UriInterface|\chillerlan\HTTP\Psr7\Uri
643
	 */
644
	public function withoutQueryValue($key):Uri{
645
		$current = $this->getQuery();
646
647
		if($current === ''){
648
			return $this;
649
		}
650
651
		$decodedKey = rawurldecode($key);
652
653
		$result = array_filter(explode('&', $current), function($part) use ($decodedKey){
654
			return rawurldecode(explode('=', $part)[0]) !== $decodedKey;
655
		});
656
657
		/** @noinspection PhpIncompatibleReturnTypeInspection */
658
		return $this->withQuery(implode('&', $result));
659
	}
660
661
	/**
662
	 * adds a specific query string value.
663
	 *
664
	 * Any existing query string values that exactly match the provided key are
665
	 * removed and replaced with the given key value pair.
666
	 *
667
	 * A value of null will set the query string key without a value, e.g. "key"
668
	 * instead of "key=value".
669
	 *
670
	 * @param string      $key   Key to set.
671
	 * @param string|null $value Value to set
672
	 *
673
	 * @return \Psr\Http\Message\UriInterface|\chillerlan\HTTP\Psr7\Uri
674
	 */
675
	public function withQueryValue($key, $value):Uri{
676
		$current = $this->getQuery();
677
678
		if($current === ''){
679
			$result = [];
680
		}
681
		else{
682
			$decodedKey = rawurldecode($key);
683
			$result     = array_filter(explode('&', $current), function($part) use ($decodedKey){
684
				return rawurldecode(explode('=', $part)[0]) !== $decodedKey;
685
			});
686
		}
687
688
		// Query string separators ("=", "&") within the key or value need to be encoded
689
		// (while preventing double-encoding) before setting the query string. All other
690
		// chars that need percent-encoding will be encoded by withQuery().
691
		$replaceQuery = ['=' => '%3D', '&' => '%26'];
692
		$key          = strtr($key, $replaceQuery);
693
694
		$result[] = $value !== null
695
			? $key.'='.strtr($value, $replaceQuery)
696
			: $key;
697
698
		/** @noinspection PhpIncompatibleReturnTypeInspection */
699
		return $this->withQuery(implode('&', $result));
700
	}
701
702
}
703