Completed
Pull Request — master (#1)
by Thibaud
02:49
created

ResourceUri::fromProtocolAndResource()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 4
Code Lines 2

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 2
CRAP Score 1

Importance

Changes 1
Bugs 0 Features 0
Metric Value
c 1
b 0
f 0
dl 0
loc 4
ccs 2
cts 2
cp 1
rs 10
cc 1
eloc 2
nc 1
nop 2
crap 1
1
<?php
2
3
/*
4
 * This file is part of alchemy/resource-component.
5
 *
6
 * (c) Alchemy <[email protected]>
7
 *
8
 * For the full copyright and license information, please view the LICENSE
9
 * file that was distributed with this source code.
10
 */
11
12
namespace Alchemy\Resource;
13
14
final class ResourceUri
15
{
16
17
    const DEFAULT_PROTOCOL = 'file';
18
19
    /**
20
     * @param $uri
21
     * @return array
22
     */
23
    private static function getNonEmptyParts($uri)
24
    {
25 212
        $nonEmptyStringFilter = function ($value) {
26 212
            return $value != '';
27 212
        };
28
29 212
        return array_filter(explode(self::PROTOCOL_SEPARATOR, $uri, 2), $nonEmptyStringFilter);
30
    }
31
32
    const PROTOCOL_SEPARATOR = '://';
33
34
    /**
35
     * @param $parts
36
     * @return bool
37
     */
38 212
    private static function validateResourceParts($parts)
39
    {
40 212
        return count($parts) === 2;
41
    }
42
43
    /**
44
     * @param $protocol
45
     * @param $resource
46
     * @return self
47
     */
48 12
    public static function fromProtocolAndResource($protocol, $resource)
49
    {
50 12
        return new self($protocol . self::PROTOCOL_SEPARATOR . $resource);
51
    }
52
53
    /**
54
     * @param $uri
55
     * @return bool
56
     */
57 224
    public static function isValidUri($uri)
58
    {
59 224
        if (strpos($uri, self::PROTOCOL_SEPARATOR) === false) {
60 12
            return false;
61
        }
62
63 212
        $parts = self::getNonEmptyParts($uri);
64
65 212
        if (! self::validateResourceParts($parts)) {
66 44
            return false;
67
        }
68
69 188
        if (strpos($parts[1], self::PROTOCOL_SEPARATOR) !== false) {
70 44
            return self::isValidUri($parts[1]);
71
        }
72
73 172
        return true;
74
    }
75
76 136
    public static function fromString($uri)
77
    {
78 136
        if (strpos($uri, self::PROTOCOL_SEPARATOR) === false) {
79 28
            $uri = self::DEFAULT_PROTOCOL . self::PROTOCOL_SEPARATOR . $uri;
80 21
        }
81
82 136
        return new self($uri);
83
    }
84
85 8
    public static function fromStringArray(array $uris)
86
    {
87 8
        $resourceUris = [];
88
89 8
        foreach ($uris as $uri) {
90 8
            $resourceUris[] = self::fromString($uri);
91 6
        }
92
93
94 4
        return $resourceUris;
95
    }
96
97
    /**
98
     * @var string
99
     */
100
    private $uri;
101
102
    /**
103
     * @var string
104
     */
105
    private $protocol;
106
107
    /**
108
     * @var string
109
     */
110
    private $resource;
111
112
    /**
113
     * @param string $uri
114
     */
115 168
    public function __construct($uri)
116
    {
117 168
        if (! self::isValidUri($uri)) {
118 24
            throw new \InvalidArgumentException(sprintf(
119 24
                'Malformed URI: required format is "protocol://resource", got "%s"',
120
                $uri
121 18
            ));
122
        }
123
124 148
        $this->uri = $uri;
125
126 148
        list ($this->protocol, $this->resource) = explode('://', $uri, 2);
127 148
    }
128
129
    /**
130
     * @return string
131
     */
132 56
    public function getUri()
133
    {
134 56
        return $this->uri;
135
    }
136
137
    /**
138
     * @return string
139
     */
140 28
    public function getProtocol()
141
    {
142 28
        return $this->protocol;
143
    }
144
145
    /**
146
     * @return string
147
     */
148 12
    public function getResource()
149
    {
150 12
        return $this->resource;
151
    }
152
153
    /**
154
     * @return bool
155
     */
156 12
    public function hasChainedResource()
157
    {
158 12
        return self::isValidUri($this->resource);
159
    }
160
161
    /**
162
     * @return self
163
     */
164 12
    public function getChainedResource()
165
    {
166 12
        return new self($this->resource);
167
    }
168
169
    /**
170
     * @param ResourceUri $other
171
     * @return bool
172
     */
173 8
    public function equals(ResourceUri $other)
174
    {
175 8
        return $this->getUri() == $other->getUri();
176
    }
177
178 4
    public function chain($containerProtocol)
179
    {
180 4
        return self::fromProtocolAndResource($containerProtocol, (string) $this);
181
    }
182
183 4
    public function child($childRelativePath)
184
    {
185 4
        return self::fromProtocolAndResource($this->protocol, $this->resource . '/' . ltrim($childRelativePath, '/'));
186
    }
187
188
    /**
189
     * @return string
190
     */
191 40
    public function __toString()
192
    {
193 40
        return $this->uri;
194
    }
195
}
196