File Coverage

blib/lib/Mojo/Path.pm
Criterion Covered Total %
statement 82 82 100.0
branch 40 40 100.0
condition 16 17 94.1
subroutine 20 20 100.0
pod 13 13 100.0
total 171 172 99.4


line stmt bran cond sub pod time code
1             package Mojo::Path;
2 63     63   63326 use Mojo::Base -base;
  63         145  
  63         464  
3 63     63   518 use overload '@{}' => sub { shift->parts }, bool => sub {1}, '""' => sub { shift->to_string }, fallback => 1;
  63     6502   182  
  63     6345   926  
  3     4136   23  
  18174         39860  
  259         26198  
4              
5 63     63   7222 use Mojo::Util qw(decode encode url_escape url_unescape);
  63         306  
  63         104318  
6              
7             has charset => 'UTF-8';
8              
9             sub canonicalize {
10 932     932 1 1610 my $self = shift;
11              
12 932         2067 my $parts = $self->parts;
13 932         3601 for (my $i = 0; $i <= $#$parts;) {
14 1386 100 100     11738 if (!length $parts->[$i] || $parts->[$i] eq '.' || $parts->[$i] eq '...') { splice @$parts, $i, 1 }
  12 100 100     45  
      100        
      100        
15 1356         3268 elsif ($i < 1 || $parts->[$i] ne '..' || $parts->[$i - 1] eq '..') { $i++ }
16 18         70 else { splice @$parts, --$i, 2 }
17             }
18              
19 932 100       3369 return @$parts ? $self : $self->trailing_slash(undef);
20             }
21              
22             sub clone {
23 2631     2631 1 4182 my $self = shift;
24              
25 2631         5491 my $clone = $self->new;
26 2631 100       7272 if (exists $self->{charset}) { $clone->{charset} = $self->{charset} }
  698         1930  
27 2631 100       5952 if (my $parts = $self->{parts}) {
28 152         743 $clone->{$_} = $self->{$_} for qw(leading_slash trailing_slash);
29 152         539 $clone->{parts} = [@$parts];
30             }
31 2479         6334 else { $clone->{path} = $self->{path} }
32              
33 2631         6250 return $clone;
34             }
35              
36 246 100   246 1 1091 sub contains { $_[1] eq '/' || $_[0]->to_route =~ m!^\Q$_[1]\E(?:/|$)! }
37              
38 2044     2044 1 4680 sub leading_slash { shift->_parse(leading_slash => @_) }
39              
40             sub merge {
41 3917     3917 1 11355 my ($self, $path) = @_;
42              
43             # Replace
44 3917 100       17692 return $self->parse($path) if $path =~ m!^/!;
45              
46             # Merge
47 353 100       1262 pop @{$self->parts} unless $self->trailing_slash;
  337         1058  
48 353         1059 $path = $self->new($path);
49 353         706 push @{$self->parts}, @{$path->parts};
  353         983  
  353         811  
50 353         1239 return $self->trailing_slash($path->trailing_slash);
51             }
52              
53 7882 100   7882 1 71034 sub new { @_ > 1 ? shift->SUPER::new->parse(@_) : shift->SUPER::new }
54              
55             sub parse {
56 4383     4383 1 7243 my $self = shift;
57 4383         10730 $self->{path} = shift;
58 4383         11117 delete @$self{qw(leading_slash parts trailing_slash)};
59 4383         11083 return $self;
60             }
61              
62 7835     7835 1 16680 sub parts { shift->_parse(parts => @_) }
63              
64             sub to_abs_string {
65 821     821 1 2007 my $path = shift->to_string;
66 821 100       3993 return $path =~ m!^/! ? $path : "/$path";
67             }
68              
69             sub to_dir {
70 69     69 1 238 my $clone = shift->clone;
71 69 100       170 pop @{$clone->parts} unless $clone->trailing_slash;
  67         195  
72 69         145 return $clone->trailing_slash(!!@{$clone->parts});
  69         144  
73             }
74              
75             sub to_route {
76 1198     1198 1 3266 my $clone = shift->clone;
77 1198 100       2205 return '/' . join '/', @{$clone->parts}, $clone->trailing_slash ? '' : ();
  1198         2422  
78             }
79              
80             sub to_string {
81 3692     3692 1 5988 my $self = shift;
82              
83             # Path
84 3692         8454 my $charset = $self->charset;
85 3692 100       10523 if (defined(my $path = $self->{path})) {
86 1935 100       7029 $path = encode $charset, $path if $charset;
87 1935         6732 return url_escape $path, '^A-Za-z0-9\-._~!$&\'()*+,;=%:@/';
88             }
89              
90             # Build path
91 1757         2772 my @parts = @{$self->parts};
  1757         3386  
92 1757 100       4957 @parts = map { encode $charset, $_ } @parts if $charset;
  3049         6530  
93 1757         3683 my $path = join '/', map { url_escape $_, '^A-Za-z0-9\-._~!$&\'()*+,;=:@' } @parts;
  3052         6457  
94 1757 100       4602 $path = "/$path" if $self->leading_slash;
95 1757 100       4123 $path = "$path/" if $self->trailing_slash;
96 1757         8103 return $path;
97             }
98              
99 4986     4986 1 10431 sub trailing_slash { shift->_parse(trailing_slash => @_) }
100              
101             sub _parse {
102 14865     14865   26148 my ($self, $name) = (shift, shift);
103              
104 14865 100       32289 unless ($self->{parts}) {
105 4424   100     17580 my $path = url_unescape delete($self->{path}) // '';
106 4424         11343 my $charset = $self->charset;
107 4424 100 66     14954 $path = decode($charset, $path) // $path if $charset;
108 4424         22448 $self->{leading_slash} = $path =~ s!^/!!;
109 4424         12890 $self->{trailing_slash} = $path =~ s!/$!!;
110 4424         17115 $self->{parts} = [split /\//, $path, -1];
111             }
112              
113 14865 100       62522 return $self->{$name} unless @_;
114 1839         3848 $self->{$name} = shift;
115 1839         5330 return $self;
116             }
117              
118             1;
119              
120             =encoding utf8
121              
122             =head1 NAME
123              
124             Mojo::Path - Path
125              
126             =head1 SYNOPSIS
127              
128             use Mojo::Path;
129              
130             # Parse
131             my $path = Mojo::Path->new('/foo%2Fbar%3B/baz.html');
132             say $path->[0];
133              
134             # Build
135             my $path = Mojo::Path->new('/i/♥');
136             push @$path, 'mojolicious';
137             say "$path";
138              
139             =head1 DESCRIPTION
140              
141             L is a container for paths used by L, based on L.
142              
143             =head1 ATTRIBUTES
144              
145             L implements the following attributes.
146              
147             =head2 charset
148              
149             my $charset = $path->charset;
150             $path = $path->charset('UTF-8');
151              
152             Charset used for encoding and decoding, defaults to C.
153              
154             # Disable encoding and decoding
155             $path->charset(undef);
156              
157             =head1 METHODS
158              
159             L inherits all methods from L and implements the following new ones.
160              
161             =head2 canonicalize
162              
163             $path = $path->canonicalize;
164              
165             Canonicalize path by resolving C<.> and C<..>, in addition C<...> will be treated as C<.> to protect from path
166             traversal attacks.
167              
168             # "/foo/baz"
169             Mojo::Path->new('/foo/./bar/../baz')->canonicalize;
170              
171             # "/../baz"
172             Mojo::Path->new('/foo/../bar/../../baz')->canonicalize;
173              
174             # "/foo/bar"
175             Mojo::Path->new('/foo/.../bar')->canonicalize;
176              
177             =head2 clone
178              
179             my $clone = $path->clone;
180              
181             Return a new L object cloned from this path.
182              
183             =head2 contains
184              
185             my $bool = $path->contains('/i/♥/mojolicious');
186              
187             Check if path contains given prefix.
188              
189             # True
190             Mojo::Path->new('/foo/bar')->contains('/');
191             Mojo::Path->new('/foo/bar')->contains('/foo');
192             Mojo::Path->new('/foo/bar')->contains('/foo/bar');
193              
194             # False
195             Mojo::Path->new('/foo/bar')->contains('/f');
196             Mojo::Path->new('/foo/bar')->contains('/bar');
197             Mojo::Path->new('/foo/bar')->contains('/whatever');
198              
199             =head2 leading_slash
200              
201             my $bool = $path->leading_slash;
202             $path = $path->leading_slash($bool);
203              
204             Path has a leading slash. Note that this method will normalize the path and that C<%2F> will be treated as C for
205             security reasons.
206              
207             # "/foo/bar"
208             Mojo::Path->new('foo/bar')->leading_slash(1);
209              
210             # "foo/bar"
211             Mojo::Path->new('/foo/bar')->leading_slash(0);
212              
213             =head2 merge
214              
215             $path = $path->merge('/foo/bar');
216             $path = $path->merge('foo/bar');
217             $path = $path->merge(Mojo::Path->new);
218              
219             Merge paths. Note that this method will normalize both paths if necessary and that C<%2F> will be treated as C for
220             security reasons.
221              
222             # "/baz/yada"
223             Mojo::Path->new('/foo/bar')->merge('/baz/yada');
224              
225             # "/foo/baz/yada"
226             Mojo::Path->new('/foo/bar')->merge('baz/yada');
227              
228             # "/foo/bar/baz/yada"
229             Mojo::Path->new('/foo/bar/')->merge('baz/yada');
230              
231             =head2 new
232              
233             my $path = Mojo::Path->new;
234             my $path = Mojo::Path->new('/foo%2Fbar%3B/baz.html');
235              
236             Construct a new L object and L path if necessary.
237              
238             =head2 parse
239              
240             $path = $path->parse('/foo%2Fbar%3B/baz.html');
241              
242             Parse path.
243              
244             =head2 to_abs_string
245              
246             my $str = $path->to_abs_string;
247              
248             Turn path into an absolute string.
249              
250             # "/i/%E2%99%A5/mojolicious"
251             Mojo::Path->new('/i/%E2%99%A5/mojolicious')->to_abs_string;
252             Mojo::Path->new('i/%E2%99%A5/mojolicious')->to_abs_string;
253              
254             =head2 parts
255              
256             my $parts = $path->parts;
257             $path = $path->parts([qw(foo bar baz)]);
258              
259             The path parts. Note that this method will normalize the path and that C<%2F> will be treated as C for security
260             reasons.
261              
262             # Part with slash
263             push @{$path->parts}, 'foo/bar';
264              
265             =head2 to_dir
266              
267             my $dir = $route->to_dir;
268              
269             Clone path and remove everything after the right-most slash.
270              
271             # "/i/%E2%99%A5/"
272             Mojo::Path->new('/i/%E2%99%A5/mojolicious')->to_dir->to_abs_string;
273              
274             # "i/%E2%99%A5/"
275             Mojo::Path->new('i/%E2%99%A5/mojolicious')->to_dir->to_abs_string;
276              
277             =head2 to_route
278              
279             my $route = $path->to_route;
280              
281             Turn path into a route.
282              
283             # "/i/♥/mojolicious"
284             Mojo::Path->new('/i/%E2%99%A5/mojolicious')->to_route;
285             Mojo::Path->new('i/%E2%99%A5/mojolicious')->to_route;
286              
287             =head2 to_string
288              
289             my $str = $path->to_string;
290              
291             Turn path into a string.
292              
293             # "/i/%E2%99%A5/mojolicious"
294             Mojo::Path->new('/i/%E2%99%A5/mojolicious')->to_string;
295              
296             # "i/%E2%99%A5/mojolicious"
297             Mojo::Path->new('i/%E2%99%A5/mojolicious')->to_string;
298              
299             =head2 trailing_slash
300              
301             my $bool = $path->trailing_slash;
302             $path = $path->trailing_slash($bool);
303              
304             Path has a trailing slash. Note that this method will normalize the path and that C<%2F> will be treated as C for
305             security reasons.
306              
307             # "/foo/bar/"
308             Mojo::Path->new('/foo/bar')->trailing_slash(1);
309              
310             # "/foo/bar"
311             Mojo::Path->new('/foo/bar/')->trailing_slash(0);
312              
313             =head1 OPERATORS
314              
315             L overloads the following operators.
316              
317             =head2 array
318              
319             my @parts = @$path;
320              
321             Alias for L. Note that this will normalize the path and that C<%2F> will be treated as C for security
322             reasons.
323              
324             say $path->[0];
325             say for @$path;
326              
327             =head2 bool
328              
329             my $bool = !!$path;
330              
331             Always true.
332              
333             =head2 stringify
334              
335             my $str = "$path";
336              
337             Alias for L.
338              
339             =head1 SEE ALSO
340              
341             L, L, L.
342              
343             =cut