line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
package CatalystX::QueryModel::DoesQueryModel; |
2
|
|
|
|
|
|
|
|
3
|
6
|
|
|
6
|
|
2995
|
use Moo::Role; |
|
6
|
|
|
|
|
20
|
|
|
6
|
|
|
|
|
55
|
|
4
|
6
|
|
|
6
|
|
16920
|
use Scalar::Util; |
|
6
|
|
|
|
|
18
|
|
|
6
|
|
|
|
|
268
|
|
5
|
6
|
|
|
6
|
|
45
|
use CatalystX::RequestModel::Utils::BadRequest; |
|
6
|
|
|
|
|
17
|
|
|
6
|
|
|
|
|
141
|
|
6
|
6
|
|
|
6
|
|
2724
|
use CatalystX::QueryModel::QueryParser; |
|
6
|
|
|
|
|
32
|
|
|
6
|
|
|
|
|
7857
|
|
7
|
|
|
|
|
|
|
|
8
|
|
|
|
|
|
|
has ctx => (is=>'ro'); |
9
|
|
|
|
|
|
|
has current_namespace => (is=>'ro', predicate=>'has_current_namespace'); |
10
|
|
|
|
|
|
|
has current_parser => (is=>'ro', predicate=>'has_current_parser'); |
11
|
|
|
|
|
|
|
has catalyst_component_name => (is=>'ro'); |
12
|
|
|
|
|
|
|
|
13
|
|
|
|
|
|
|
sub namespace { |
14
|
15
|
|
|
15
|
|
54
|
my ($class_or_self, @data) = @_; |
15
|
15
|
100
|
|
|
|
63
|
my $class = ref($class_or_self) ? ref($class_or_self) : $class_or_self; |
16
|
15
|
100
|
|
|
|
52
|
if(@data) { |
17
|
6
|
|
|
|
|
17
|
@data = map { split /\./, $_ } @data; |
|
6
|
|
|
|
|
36
|
|
18
|
6
|
|
|
|
|
27
|
CatalystX::QueryModel::_add_metadata($class, 'namespace', @data); |
19
|
|
|
|
|
|
|
} |
20
|
|
|
|
|
|
|
|
21
|
15
|
100
|
|
|
|
116
|
return $class_or_self->namespace_metadata if $class_or_self->can('namespace_metadata'); |
22
|
|
|
|
|
|
|
} |
23
|
|
|
|
|
|
|
|
24
|
|
|
|
|
|
|
sub has_content_type { |
25
|
9
|
|
|
9
|
0
|
154
|
my ($self) = @_; |
26
|
9
|
50
|
|
|
|
41
|
return $self->content_type ? 1:0; |
27
|
|
|
|
|
|
|
} |
28
|
|
|
|
|
|
|
|
29
|
|
|
|
|
|
|
sub content_type { |
30
|
9
|
|
|
9
|
|
21
|
my ($class_or_self, $ct) = @_; |
31
|
9
|
50
|
|
|
|
24
|
my $class = ref($class_or_self) ? ref($class_or_self) : $class_or_self; |
32
|
9
|
50
|
|
|
|
24
|
CatalystX::QueryModel::_add_metadata($class, 'content_type', $ct) if $ct; |
33
|
|
|
|
|
|
|
|
34
|
9
|
50
|
|
|
|
98
|
if($class_or_self->can('content_type_metadata')) { |
35
|
0
|
|
|
|
|
0
|
my ($ct) = $class_or_self->content_type_metadata; # needed because this returns an array but we only want the first one |
36
|
0
|
|
|
|
|
0
|
return $ct; |
37
|
|
|
|
|
|
|
} |
38
|
|
|
|
|
|
|
} |
39
|
|
|
|
|
|
|
|
40
|
|
|
|
|
|
|
sub property { |
41
|
60
|
|
|
60
|
|
161
|
my ($class_or_self, $attr, $data_proto, $options) = @_; |
42
|
60
|
50
|
|
|
|
149
|
my $class = ref($class_or_self) ? ref($class_or_self) : $class_or_self; |
43
|
60
|
50
|
|
|
|
144
|
if(defined $data_proto) { |
44
|
60
|
50
|
50
|
|
|
209
|
my $data = (ref($data_proto)||'') eq 'HASH' ? $data_proto : +{ name => $attr }; |
45
|
60
|
50
|
|
|
|
139
|
$data->{name} = $attr unless exists($data->{name}); |
46
|
60
|
|
|
|
|
213
|
CatalystX::QueryModel::_add_metadata($class, 'property_data', +{$attr => $data}); |
47
|
|
|
|
|
|
|
} |
48
|
|
|
|
|
|
|
} |
49
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
sub properties { |
51
|
18
|
|
|
18
|
|
40
|
my ($class_or_self, @data) = @_; |
52
|
18
|
50
|
|
|
|
50
|
my $class = ref($class_or_self) ? ref($class_or_self) : $class_or_self; |
53
|
18
|
|
|
|
|
54
|
while(@data) { |
54
|
0
|
|
|
|
|
0
|
my $attr = shift(@data); |
55
|
0
|
0
|
0
|
|
|
0
|
my $data = (ref($data[0])||'') eq 'HASH' ? shift(@data) : +{ name => $attr }; |
56
|
0
|
0
|
|
|
|
0
|
$data->{name} = $attr unless exists($data->{name}); |
57
|
0
|
|
|
|
|
0
|
CatalystX::QueryModel::_add_metadata($class, 'property_data', +{$attr => $data}); |
58
|
|
|
|
|
|
|
} |
59
|
|
|
|
|
|
|
|
60
|
18
|
50
|
|
|
|
126
|
return $class_or_self->property_data_metadata if $class_or_self->can('property_data_metadata'); |
61
|
|
|
|
|
|
|
} |
62
|
|
|
|
|
|
|
|
63
|
|
|
|
|
|
|
sub COMPONENT { |
64
|
24
|
|
|
24
|
0
|
6536
|
my ($class, $app, $args) = @_; |
65
|
24
|
|
|
|
|
202
|
$args = $class->merge_config_hashes($class->config, $args); |
66
|
24
|
|
|
|
|
4570
|
return bless $args, $class; |
67
|
|
|
|
|
|
|
} |
68
|
|
|
|
|
|
|
|
69
|
|
|
|
|
|
|
## TODO handle if we are wrapping a model that already does ACCEPT_CONTEXT |
70
|
|
|
|
|
|
|
sub ACCEPT_CONTEXT { |
71
|
9
|
|
|
9
|
0
|
586
|
my $self = shift; |
72
|
9
|
|
|
|
|
16
|
my $c = shift; |
73
|
|
|
|
|
|
|
|
74
|
9
|
|
|
|
|
89
|
my %args = (%$self, @_); |
75
|
9
|
|
|
|
|
51
|
my %query_args = $self->parse_query($c, %args); |
76
|
9
|
|
|
|
|
43
|
my %init_args = (%args, %query_args, ctx=>$c); |
77
|
9
|
|
|
|
|
23
|
my $class = ref($self); |
78
|
|
|
|
|
|
|
|
79
|
9
|
|
|
|
|
39
|
return my $request_model = $self->build_query_model($c, $class, %init_args); |
80
|
|
|
|
|
|
|
} |
81
|
|
|
|
|
|
|
|
82
|
|
|
|
|
|
|
sub build_query_model { |
83
|
9
|
|
|
9
|
0
|
34
|
my ($self, $c, $class, %init_args) = @_; |
84
|
|
|
|
|
|
|
|
85
|
|
|
|
|
|
|
my $request_model = eval { |
86
|
|
|
|
|
|
|
$class->new(%init_args) |
87
|
9
|
|
33
|
|
|
17
|
} || do { |
88
|
|
|
|
|
|
|
CatalystX::RequestModel::Utils::BadRequest->throw(class=>$class, error_trace=>$@); |
89
|
|
|
|
|
|
|
}; |
90
|
9
|
|
|
|
|
2641
|
return $request_model; |
91
|
|
|
|
|
|
|
} |
92
|
|
|
|
|
|
|
|
93
|
|
|
|
|
|
|
sub parse_query { |
94
|
9
|
|
|
9
|
0
|
34
|
my ($self, $c, %args) = @_; |
95
|
|
|
|
|
|
|
|
96
|
9
|
|
|
|
|
48
|
my @rules = $self->properties; |
97
|
9
|
|
|
|
|
33
|
my @ns = $self->get_namespace(%args); |
98
|
9
|
|
|
|
|
21
|
my $parser_class = 'CatalystX::QueryModel::QueryParser'; |
99
|
|
|
|
|
|
|
my $parser = exists($args{current_parser}) ? |
100
|
|
|
|
|
|
|
$args{current_parser} : |
101
|
9
|
50
|
|
|
|
81
|
$parser_class->new(ctx=>$c, request_model=>$self ); |
102
|
|
|
|
|
|
|
|
103
|
9
|
50
|
|
|
|
27
|
$parser->{context} = $args{context} if exists $args{context}; ## TODO ulgy |
104
|
9
|
|
|
|
|
65
|
my %request_args = $parser->parse(\@ns, \@rules); |
105
|
9
|
|
|
|
|
63
|
return %request_args; |
106
|
|
|
|
|
|
|
} |
107
|
|
|
|
|
|
|
|
108
|
|
|
|
|
|
|
sub get_content_type { |
109
|
0
|
|
|
0
|
0
|
0
|
my ($self, $c) = @_; |
110
|
0
|
|
|
|
|
0
|
my $ct = $c->req->content_type; |
111
|
0
|
0
|
0
|
|
|
0
|
return 'application/x-www-form-urlencoded' if !$ct && $c->req->method eq 'GET'; |
112
|
0
|
0
|
|
|
|
0
|
return 'application/x-www-form-urlencoded' if $self->get_content_in eq 'query'; |
113
|
0
|
|
|
|
|
0
|
return $ct; |
114
|
|
|
|
|
|
|
} |
115
|
|
|
|
|
|
|
|
116
|
|
|
|
|
|
|
sub get_namespace { |
117
|
9
|
|
|
9
|
0
|
26
|
my ($self, %args) = @_; |
118
|
9
|
50
|
|
|
|
27
|
return @{$args{current_namespace}} if exists($args{current_namespace}); |
|
0
|
|
|
|
|
0
|
|
119
|
9
|
|
|
|
|
36
|
return grep { defined $_ } $self->namespace; |
|
9
|
|
|
|
|
40
|
|
120
|
|
|
|
|
|
|
} |
121
|
|
|
|
|
|
|
|
122
|
|
|
|
|
|
|
sub get_attribute_value_for { |
123
|
17
|
|
|
17
|
0
|
38
|
my ($self, $attr) = @_; |
124
|
17
|
|
|
|
|
669
|
return $self->$attr; |
125
|
|
|
|
|
|
|
} |
126
|
|
|
|
|
|
|
|
127
|
|
|
|
|
|
|
sub nested_params { |
128
|
9
|
|
|
9
|
1
|
682
|
my $self = shift; |
129
|
9
|
|
|
|
|
18
|
my %return; |
130
|
9
|
|
|
|
|
30
|
foreach my $p ($self->properties) { |
131
|
23
|
|
|
|
|
77
|
my ($attr, $meta) = %$p; |
132
|
23
|
100
|
|
|
|
62
|
if(my $predicate = $meta->{attr_predicate}) { |
133
|
19
|
50
|
|
|
|
48
|
if($meta->{omit_empty}) { |
134
|
19
|
100
|
|
|
|
708
|
next unless $self->$predicate; # skip empties when omit_empty=>1 |
135
|
|
|
|
|
|
|
} |
136
|
|
|
|
|
|
|
} |
137
|
|
|
|
|
|
|
|
138
|
17
|
|
|
|
|
52
|
my $value = $self->get_attribute_value_for($attr); |
139
|
17
|
50
|
50
|
|
|
114
|
if( (ref($value)||'') eq 'ARRAY') { |
|
|
50
|
33
|
|
|
|
|
140
|
0
|
|
|
|
|
0
|
my @gathered = (); |
141
|
0
|
|
|
|
|
0
|
foreach my $v (@$value) { |
142
|
0
|
0
|
|
|
|
0
|
if(Scalar::Util::blessed($v)) { |
143
|
0
|
|
|
|
|
0
|
my $params = $v->nested_params; |
144
|
0
|
0
|
|
|
|
0
|
push @gathered, $params if keys(%$params); |
145
|
|
|
|
|
|
|
} else { |
146
|
0
|
|
|
|
|
0
|
push @gathered, $v; |
147
|
|
|
|
|
|
|
} |
148
|
|
|
|
|
|
|
|
149
|
|
|
|
|
|
|
} |
150
|
0
|
|
|
|
|
0
|
$return{$attr} = \@gathered; |
151
|
|
|
|
|
|
|
} elsif(Scalar::Util::blessed($value) && $value->can('nested_params')) { |
152
|
0
|
|
|
|
|
0
|
my $params = $value->nested_params; |
153
|
0
|
0
|
|
|
|
0
|
next unless keys(%$params); |
154
|
0
|
|
|
|
|
0
|
$return{$attr} = $params; |
155
|
|
|
|
|
|
|
} else { |
156
|
17
|
|
|
|
|
52
|
$return{$attr} = $value; |
157
|
|
|
|
|
|
|
} |
158
|
|
|
|
|
|
|
} |
159
|
9
|
|
|
|
|
54
|
return \%return; |
160
|
|
|
|
|
|
|
} |
161
|
|
|
|
|
|
|
|
162
|
|
|
|
|
|
|
sub get { |
163
|
0
|
|
|
0
|
1
|
|
my ($self, @fields) = @_; |
164
|
0
|
|
|
|
|
|
my $p = $self->nested_params; |
165
|
0
|
|
|
|
|
|
my @got = @$p{@fields}; |
166
|
0
|
|
|
|
|
|
return @got; |
167
|
|
|
|
|
|
|
} |
168
|
|
|
|
|
|
|
|
169
|
|
|
|
|
|
|
1; |
170
|
|
|
|
|
|
|
|
171
|
|
|
|
|
|
|
=head1 NAME |
172
|
|
|
|
|
|
|
|
173
|
|
|
|
|
|
|
CatalystX::QueryModel::DoesQueryModel - Role to provide query model API |
174
|
|
|
|
|
|
|
|
175
|
|
|
|
|
|
|
=head1 SYNOPSIS |
176
|
|
|
|
|
|
|
|
177
|
|
|
|
|
|
|
Generally you will apply this role via L<CatalystX::QueryModel> |
178
|
|
|
|
|
|
|
|
179
|
|
|
|
|
|
|
See L<CatalystX::QueryModel> for a more general overview. |
180
|
|
|
|
|
|
|
|
181
|
|
|
|
|
|
|
=head1 DESCRIPTION |
182
|
|
|
|
|
|
|
|
183
|
|
|
|
|
|
|
A role that gives a L<Catalyst::Model> the ability to indicate which of its attributes should be |
184
|
|
|
|
|
|
|
consider query model data, as well as additional need meta data so that we can process it |
185
|
|
|
|
|
|
|
properly. |
186
|
|
|
|
|
|
|
|
187
|
|
|
|
|
|
|
Since we need to wrap C<has> you should never apply this role manually but rather instead use |
188
|
|
|
|
|
|
|
L<CatalystX::QueryModel> to apply it for you. If you need to customize this role you will |
189
|
|
|
|
|
|
|
also need to subclass L<CatalystX::QueryModel> and have that new subclass apply you custom |
190
|
|
|
|
|
|
|
role. Please ping me if you really need this since I guess we could change L<CatalystX::QueryModel> |
191
|
|
|
|
|
|
|
to make it easier to supply a custom role, just let me know your use case. |
192
|
|
|
|
|
|
|
|
193
|
|
|
|
|
|
|
=head1 METHODS |
194
|
|
|
|
|
|
|
|
195
|
|
|
|
|
|
|
This class defines the following public API |
196
|
|
|
|
|
|
|
|
197
|
|
|
|
|
|
|
=head2 nested_params |
198
|
|
|
|
|
|
|
|
199
|
|
|
|
|
|
|
Returns all the attributes marked as request properties in the form of a hashref. If any of the |
200
|
|
|
|
|
|
|
properties refer to an array or indexed value, or an object, we automatically follow that to |
201
|
|
|
|
|
|
|
return all the property data below. |
202
|
|
|
|
|
|
|
|
203
|
|
|
|
|
|
|
Attributes that are empty will be left out of the return data structure. |
204
|
|
|
|
|
|
|
|
205
|
|
|
|
|
|
|
Easiest way to get all your data but then again you get a structure that is very tightly tied to |
206
|
|
|
|
|
|
|
your request model. |
207
|
|
|
|
|
|
|
|
208
|
|
|
|
|
|
|
=head2 get |
209
|
|
|
|
|
|
|
|
210
|
|
|
|
|
|
|
Accepts a list of attributes that refer to request properties and returns their values. In the case |
211
|
|
|
|
|
|
|
when the attribute listed has no value, you will instead get an C<undef>. |
212
|
|
|
|
|
|
|
|
213
|
|
|
|
|
|
|
=head1 EXCEPTIONS |
214
|
|
|
|
|
|
|
|
215
|
|
|
|
|
|
|
This class can throw the following exceptions: |
216
|
|
|
|
|
|
|
|
217
|
|
|
|
|
|
|
=head2 Invalid Request Content Body |
218
|
|
|
|
|
|
|
|
219
|
|
|
|
|
|
|
If we can't create an instance of the request model we throw a L<CatalystX::RequestModel::Utils::BadRequest>. |
220
|
|
|
|
|
|
|
This will get interpretated as an HTTP 400 status client error if you are using L<CatalystX::Errors>. |
221
|
|
|
|
|
|
|
|
222
|
|
|
|
|
|
|
=head1 AUTHOR |
223
|
|
|
|
|
|
|
|
224
|
|
|
|
|
|
|
See L<CatalystX::QueryModel>. |
225
|
|
|
|
|
|
|
|
226
|
|
|
|
|
|
|
=head1 COPYRIGHT |
227
|
|
|
|
|
|
|
|
228
|
|
|
|
|
|
|
See L<CatalystX::QueryModel>. |
229
|
|
|
|
|
|
|
|
230
|
|
|
|
|
|
|
=head1 LICENSE |
231
|
|
|
|
|
|
|
|
232
|
|
|
|
|
|
|
See L<CatalystX::QueryModel>. |
233
|
|
|
|
|
|
|
|
234
|
|
|
|
|
|
|
=cut |
235
|
|
|
|
|
|
|
|