File Coverage

blib/lib/Mojo/Parameters.pm
Criterion Covered Total %
statement 107 107 100.0
branch 50 52 96.1
condition 9 12 75.0
subroutine 18 18 100.0
pod 12 12 100.0
total 196 201 97.5


line stmt bran cond sub pod time code
1             package Mojo::Parameters;
2 64     64   68446 use Mojo::Base -base;
  64         132  
  64         526  
3 64     64   762 use overload '@{}' => sub { shift->pairs }, bool => sub {1}, '""' => sub { shift->to_string }, fallback => 1;
  64     2870   219  
  64     1981   835  
  2     2700   8  
  7593         23562  
  66         11387  
4              
5 64     64   7259 use Mojo::Util qw(decode encode url_escape url_unescape);
  64         167  
  64         126497  
6              
7             has charset => 'UTF-8';
8              
9             sub append {
10 460     460 1 910 my $self = shift;
11              
12 460         1044 my $old = $self->pairs;
13 460 100       1580 my @new = @_ == 1 ? @{shift->pairs} : @_;
  315         766  
14 460         2336 while (my ($name, $value) = splice @new, 0, 2) {
15              
16             # Multiple values
17 336 100 50     1102 if (ref $value eq 'ARRAY') { push @$old, $name => $_ // '' for @$value }
  48 100       357  
18              
19             # Single value
20 286         1126 elsif (defined $value) { push @$old, $name => $value }
21             }
22              
23 460         1454 return $self;
24             }
25              
26             sub clone {
27 1584     1584 1 2860 my $self = shift;
28              
29 1584         3390 my $clone = $self->new;
30 1584 100       4713 if (exists $self->{charset}) { $clone->{charset} = $self->{charset} }
  452         1255  
31 1584 100       3958 if (defined $self->{string}) { $clone->{string} = $self->{string} }
  193         687  
32 1391         2151 else { $clone->{pairs} = [@{$self->pairs}] }
  1391         3137  
33              
34 1584         5319 return $clone;
35             }
36              
37             sub every_param {
38 2810     2810 1 5448 my ($self, $name) = @_;
39              
40 2810         4103 my @values;
41 2810         5120 my $pairs = $self->pairs;
42 2810         7863 for (my $i = 0; $i < @$pairs; $i += 2) {
43 1175 100       3698 push @values, $pairs->[$i + 1] if $pairs->[$i] eq $name;
44             }
45              
46 2810         12796 return \@values;
47             }
48              
49             sub merge {
50 25     25 1 57 my $self = shift;
51              
52 25 100       106 my $merge = @_ == 1 ? shift->to_hash : {@_};
53 25         132 for my $name (sort keys %$merge) {
54 30         62 my $value = $merge->{$name};
55 30 100       111 defined $value ? $self->param($name => $value) : $self->remove($name);
56             }
57              
58 25         110 return $self;
59             }
60              
61 139     139 1 286 sub names { [sort keys %{shift->to_hash}] }
  139         501  
62              
63 4945 100   4945 1 72495 sub new { @_ > 1 ? shift->SUPER::new->parse(@_) : shift->SUPER::new }
64              
65             sub pairs {
66 7829     7829 1 11460 my $self = shift;
67              
68             # Replace parameters
69 7829 100       16013 if (@_) {
70 5         21 $self->{pairs} = shift;
71 5         13 delete $self->{string};
72 5         22 return $self;
73             }
74              
75             # Parse string
76 7824 100       18085 if (defined(my $str = delete $self->{string})) {
77 284         831 my $pairs = $self->{pairs} = [];
78 284 50       848 return $pairs unless length $str;
79              
80 284         1156 my $charset = $self->charset;
81 284         1232 for my $pair (split /&/, $str) {
82 441 50       2753 next unless $pair =~ /^([^=]+)(?:=(.*))?$/;
83 441   100     2112 my ($name, $value) = ($1, $2 // '');
84              
85             # Replace "+" with whitespace, unescape and decode
86 441         1547 s/\+/ /g for $name, $value;
87 441         1805 $name = url_unescape $name;
88 441 100 66     1512 $name = decode($charset, $name) // $name if $charset;
89 441         1439 $value = url_unescape $value;
90 441 100 66     1325 $value = decode($charset, $value) // $value if $charset;
91              
92 441         1450 push @$pairs, $name, $value;
93             }
94             }
95              
96 7824   100     26243 return $self->{pairs} //= [];
97             }
98              
99             sub param {
100 2533     2533 1 5592 my ($self, $name) = (shift, shift);
101 2533 100       7195 return $self->every_param($name)->[-1] unless @_;
102 35         147 $self->remove($name);
103 35 100       182 return $self->append($name => ref $_[0] eq 'ARRAY' ? $_[0] : [@_]);
104             }
105              
106             sub parse {
107 404     404 1 802 my $self = shift;
108              
109             # Pairs
110 404 100       1251 return $self->append(@_) if @_ > 1;
111              
112             # String
113 355         1127 $self->{string} = shift;
114 355         1011 return $self;
115             }
116              
117             sub remove {
118 42     42 1 98 my ($self, $name) = @_;
119 42         101 my $pairs = $self->pairs;
120 42         86 my $i = 0;
121 42 100       304 $pairs->[$i] eq $name ? splice @$pairs, $i, 2 : ($i += 2) while $i < @$pairs;
122 42         90 return $self;
123             }
124              
125             sub to_hash {
126 331     331 1 724 my $self = shift;
127              
128 331         560 my %hash;
129 331         770 my $pairs = $self->pairs;
130 331         1410 for (my $i = 0; $i < @$pairs; $i += 2) {
131 211         395 my ($name, $value) = @{$pairs}[$i, $i + 1];
  211         482  
132              
133             # Array
134 211 100       475 if (exists $hash{$name}) {
135 37 100       160 $hash{$name} = [$hash{$name}] if ref $hash{$name} ne 'ARRAY';
136 37         75 push @{$hash{$name}}, $value;
  37         143  
137             }
138              
139             # String
140 174         676 else { $hash{$name} = $value }
141             }
142              
143 331         2071 return \%hash;
144             }
145              
146             sub to_string {
147 2702     2702 1 4822 my $self = shift;
148              
149             # String (RFC 3986)
150 2702         6886 my $charset = $self->charset;
151 2702 100       8466 if (defined(my $str = $self->{string})) {
152 238 100       1747 $str = encode $charset, $str if $charset;
153 238         1217 return url_escape $str, '^A-Za-z0-9\-._~%!$&\'()*+,;=:@/?';
154             }
155              
156             # Build pairs (HTML Living Standard)
157 2464         5527 my $pairs = $self->pairs;
158 2464 100       8509 return '' unless @$pairs;
159 104         249 my @pairs;
160 104         396 for (my $i = 0; $i < @$pairs; $i += 2) {
161 216         478 my ($name, $value) = @{$pairs}[$i, $i + 1];
  216         518  
162              
163             # Escape and replace whitespace with "+"
164 216 100       770 $name = encode $charset, $name if $charset;
165 216         10558 $name = url_escape $name, '^*\-.0-9A-Z_a-z';
166 216 100       811 $value = encode $charset, $value if $charset;
167 216         547 $value = url_escape $value, '^*\-.0-9A-Z_a-z';
168 216         871 s/\%20/\+/g for $name, $value;
169              
170 216         913 push @pairs, "$name=$value";
171             }
172              
173 104         740 return join '&', @pairs;
174             }
175              
176             1;
177              
178             =encoding utf8
179              
180             =head1 NAME
181              
182             Mojo::Parameters - Parameters
183              
184             =head1 SYNOPSIS
185              
186             use Mojo::Parameters;
187              
188             # Parse
189             my $params = Mojo::Parameters->new('foo=bar&baz=23');
190             say $params->param('baz');
191              
192             # Build
193             my $params = Mojo::Parameters->new(foo => 'bar', baz => 23);
194             push @$params, i => '♥ mojolicious';
195             say "$params";
196              
197             =head1 DESCRIPTION
198              
199             L is a container for form parameters used by L, based on L
200             3986|https://tools.ietf.org/html/rfc3986> and the L.
201              
202             =head1 ATTRIBUTES
203              
204             L implements the following attributes.
205              
206             =head2 charset
207              
208             my $charset = $params->charset;
209             $params = $params->charset('UTF-8');
210              
211             Charset used for encoding and decoding parameters, defaults to C.
212              
213             # Disable encoding and decoding
214             $params->charset(undef);
215              
216             =head1 METHODS
217              
218             L inherits all methods from L and implements the following new ones.
219              
220             =head2 append
221              
222             $params = $params->append(foo => 'ba&r');
223             $params = $params->append(foo => ['ba&r', 'baz']);
224             $params = $params->append(foo => ['bar', 'baz'], bar => 23);
225             $params = $params->append(Mojo::Parameters->new);
226              
227             Append parameters. Note that this method will normalize the parameters.
228              
229             # "foo=bar&foo=baz"
230             Mojo::Parameters->new('foo=bar')->append(Mojo::Parameters->new('foo=baz'));
231              
232             # "foo=bar&foo=baz"
233             Mojo::Parameters->new('foo=bar')->append(foo => 'baz');
234              
235             # "foo=bar&foo=baz&foo=yada"
236             Mojo::Parameters->new('foo=bar')->append(foo => ['baz', 'yada']);
237              
238             # "foo=bar&foo=baz&foo=yada&bar=23"
239             Mojo::Parameters->new('foo=bar')->append(foo => ['baz', 'yada'], bar => 23);
240              
241             =head2 clone
242              
243             my $params2 = $params->clone;
244              
245             Return a new L object cloned from these parameters.
246              
247             =head2 every_param
248              
249             my $values = $params->every_param('foo');
250              
251             Similar to L, but returns all values sharing the same name as an array reference. Note that this method will
252             normalize the parameters.
253              
254             # Get first value
255             say $params->every_param('foo')->[0];
256              
257             =head2 merge
258              
259             $params = $params->merge(foo => 'ba&r');
260             $params = $params->merge(foo => ['ba&r', 'baz']);
261             $params = $params->merge(foo => ['bar', 'baz'], bar => 23);
262             $params = $params->merge(Mojo::Parameters->new);
263              
264             Merge parameters. Note that this method will normalize the parameters.
265              
266             # "foo=baz"
267             Mojo::Parameters->new('foo=bar')->merge(Mojo::Parameters->new('foo=baz'));
268              
269             # "yada=yada&foo=baz"
270             Mojo::Parameters->new('foo=bar&yada=yada')->merge(foo => 'baz');
271              
272             # "yada=yada"
273             Mojo::Parameters->new('foo=bar&yada=yada')->merge(foo => undef);
274              
275             =head2 names
276              
277             my $names = $params->names;
278              
279             Return an array reference with all parameter names.
280              
281             # Names of all parameters
282             say for @{$params->names};
283              
284             =head2 new
285              
286             my $params = Mojo::Parameters->new;
287             my $params = Mojo::Parameters->new('foo=b%3Bar&baz=23');
288             my $params = Mojo::Parameters->new(foo => 'b&ar');
289             my $params = Mojo::Parameters->new(foo => ['ba&r', 'baz']);
290             my $params = Mojo::Parameters->new(foo => ['bar', 'baz'], bar => 23);
291              
292             Construct a new L object and L parameters if necessary.
293              
294             =head2 pairs
295              
296             my $array = $params->pairs;
297             $params = $params->pairs([foo => 'b&ar', baz => 23]);
298              
299             Parsed parameter pairs. Note that this method will normalize the parameters.
300              
301             # Remove all parameters
302             $params->pairs([]);
303              
304             =head2 param
305              
306             my $value = $params->param('foo');
307             $params = $params->param(foo => 'ba&r');
308             $params = $params->param(foo => qw(ba&r baz));
309             $params = $params->param(foo => ['ba;r', 'baz']);
310              
311             Access parameter values. If there are multiple values sharing the same name, and you want to access more than just the
312             last one, you can use L. Note that this method will normalize the parameters.
313              
314             =head2 parse
315              
316             $params = $params->parse('foo=b%3Bar&baz=23');
317              
318             Parse parameters.
319              
320             =head2 remove
321              
322             $params = $params->remove('foo');
323              
324             Remove parameters. Note that this method will normalize the parameters.
325              
326             # "bar=yada"
327             Mojo::Parameters->new('foo=bar&foo=baz&bar=yada')->remove('foo');
328              
329             =head2 to_hash
330              
331             my $hash = $params->to_hash;
332              
333             Turn parameters into a hash reference. Note that this method will normalize the parameters.
334              
335             # "baz"
336             Mojo::Parameters->new('foo=bar&foo=baz')->to_hash->{foo}[1];
337              
338             =head2 to_string
339              
340             my $str = $params->to_string;
341              
342             Turn parameters into a string.
343              
344             # "foo=bar&baz=23"
345             Mojo::Parameters->new->pairs([foo => 'bar', baz => 23])->to_string;
346              
347             =head1 OPERATORS
348              
349             L overloads the following operators.
350              
351             =head2 array
352              
353             my @pairs = @$params;
354              
355             Alias for L. Note that this will normalize the parameters.
356              
357             say $params->[0];
358             say for @$params;
359              
360             =head2 bool
361              
362             my $bool = !!$params;
363              
364             Always true.
365              
366             =head2 stringify
367              
368             my $str = "$params";
369              
370             Alias for L.
371              
372             =head1 SEE ALSO
373              
374             L, L, L.
375              
376             =cut