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 63     63   64198 use Mojo::Base -base;
  63         144  
  63         532  
3 63     63   519 use overload '@{}' => sub { shift->pairs }, bool => sub {1}, '""' => sub { shift->to_string }, fallback => 1;
  63     3377   155  
  63     1879   806  
  2     2100   8  
  7337         18647  
  66         11051  
4              
5 63     63   6963 use Mojo::Util qw(decode encode url_escape url_unescape);
  63         189  
  63         114109  
6              
7             has charset => 'UTF-8';
8              
9             sub append {
10 456     456 1 931 my $self = shift;
11              
12 456         975 my $old = $self->pairs;
13 456 100       1519 my @new = @_ == 1 ? @{shift->pairs} : @_;
  311         763  
14 456         2016 while (my ($name, $value) = splice @new, 0, 2) {
15              
16             # Multiple values
17 336 100 50     1053 if (ref $value eq 'ARRAY') { push @$old, $name => $_ // '' for @$value }
  48 100       320  
18              
19             # Single value
20 286         1094 elsif (defined $value) { push @$old, $name => $value }
21             }
22              
23 456         1415 return $self;
24             }
25              
26             sub clone {
27 1573     1573 1 2853 my $self = shift;
28              
29 1573         3156 my $clone = $self->new;
30 1573 100       4645 if (exists $self->{charset}) { $clone->{charset} = $self->{charset} }
  448         1572  
31 1573 100       3825 if (defined $self->{string}) { $clone->{string} = $self->{string} }
  193         606  
32 1380         2151 else { $clone->{pairs} = [@{$self->pairs}] }
  1380         2824  
33              
34 1573         5218 return $clone;
35             }
36              
37             sub every_param {
38 2781     2781 1 5007 my ($self, $name) = @_;
39              
40 2781         3864 my @values;
41 2781         4829 my $pairs = $self->pairs;
42 2781         7940 for (my $i = 0; $i < @$pairs; $i += 2) {
43 1175 100       3436 push @values, $pairs->[$i + 1] if $pairs->[$i] eq $name;
44             }
45              
46 2781         11975 return \@values;
47             }
48              
49             sub merge {
50 25     25 1 61 my $self = shift;
51              
52 25 100       102 my $merge = @_ == 1 ? shift->to_hash : {@_};
53 25         113 for my $name (sort keys %$merge) {
54 30         61 my $value = $merge->{$name};
55 30 100       102 defined $value ? $self->param($name => $value) : $self->remove($name);
56             }
57              
58 25         88 return $self;
59             }
60              
61 137     137 1 258 sub names { [sort keys %{shift->to_hash}] }
  137         441  
62              
63 4840 100   4840 1 70870 sub new { @_ > 1 ? shift->SUPER::new->parse(@_) : shift->SUPER::new }
64              
65             sub pairs {
66 7621     7621 1 11203 my $self = shift;
67              
68             # Replace parameters
69 7621 100       15004 if (@_) {
70 5         20 $self->{pairs} = shift;
71 5         11 delete $self->{string};
72 5         21 return $self;
73             }
74              
75             # Parse string
76 7616 100       17799 if (defined(my $str = delete $self->{string})) {
77 284         764 my $pairs = $self->{pairs} = [];
78 284 50       829 return $pairs unless length $str;
79              
80 284         746 my $charset = $self->charset;
81 284         1116 for my $pair (split /&/, $str) {
82 441 50       2669 next unless $pair =~ /^([^=]+)(?:=(.*))?$/;
83 441   100     1979 my ($name, $value) = ($1, $2 // '');
84              
85             # Replace "+" with whitespace, unescape and decode
86 441         1509 s/\+/ /g for $name, $value;
87 441         1248 $name = url_unescape $name;
88 441 100 66     1471 $name = decode($charset, $name) // $name if $charset;
89 441         1053 $value = url_unescape $value;
90 441 100 66     1426 $value = decode($charset, $value) // $value if $charset;
91              
92 441         1762 push @$pairs, $name, $value;
93             }
94             }
95              
96 7616   100     25552 return $self->{pairs} //= [];
97             }
98              
99             sub param {
100 2508     2508 1 5576 my ($self, $name) = (shift, shift);
101 2508 100       6874 return $self->every_param($name)->[-1] unless @_;
102 35         134 $self->remove($name);
103 35 100       178 return $self->append($name => ref $_[0] eq 'ARRAY' ? $_[0] : [@_]);
104             }
105              
106             sub parse {
107 404     404 1 810 my $self = shift;
108              
109             # Pairs
110 404 100       1257 return $self->append(@_) if @_ > 1;
111              
112             # String
113 355         1093 $self->{string} = shift;
114 355         928 return $self;
115             }
116              
117             sub remove {
118 42     42 1 99 my ($self, $name) = @_;
119 42         101 my $pairs = $self->pairs;
120 42         94 my $i = 0;
121 42 100       271 $pairs->[$i] eq $name ? splice @$pairs, $i, 2 : ($i += 2) while $i < @$pairs;
122 42         102 return $self;
123             }
124              
125             sub to_hash {
126 321     321 1 665 my $self = shift;
127              
128 321         537 my %hash;
129 321         751 my $pairs = $self->pairs;
130 321         1182 for (my $i = 0; $i < @$pairs; $i += 2) {
131 211         338 my ($name, $value) = @{$pairs}[$i, $i + 1];
  211         420  
132              
133             # Array
134 211 100       790 if (exists $hash{$name}) {
135 37 100       122 $hash{$name} = [$hash{$name}] if ref $hash{$name} ne 'ARRAY';
136 37         56 push @{$hash{$name}}, $value;
  37         127  
137             }
138              
139             # String
140 174         617 else { $hash{$name} = $value }
141             }
142              
143 321         1866 return \%hash;
144             }
145              
146             sub to_string {
147 2552     2552 1 4455 my $self = shift;
148              
149             # String (RFC 3986)
150 2552         6198 my $charset = $self->charset;
151 2552 100       7651 if (defined(my $str = $self->{string})) {
152 238 100       1026 $str = encode $charset, $str if $charset;
153 238         1160 return url_escape $str, '^A-Za-z0-9\-._~%!$&\'()*+,;=:@/?';
154             }
155              
156             # Build pairs (HTML Living Standard)
157 2314         4951 my $pairs = $self->pairs;
158 2314 100       7902 return '' unless @$pairs;
159 104         226 my @pairs;
160 104         377 for (my $i = 0; $i < @$pairs; $i += 2) {
161 216         390 my ($name, $value) = @{$pairs}[$i, $i + 1];
  216         503  
162              
163             # Escape and replace whitespace with "+"
164 216 100       746 $name = encode $charset, $name if $charset;
165 216         8406 $name = url_escape $name, '^*\-.0-9A-Z_a-z';
166 216 100       711 $value = encode $charset, $value if $charset;
167 216         528 $value = url_escape $value, '^*\-.0-9A-Z_a-z';
168 216         794 s/\%20/\+/g for $name, $value;
169              
170 216         836 push @pairs, "$name=$value";
171             }
172              
173 104         686 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