| line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
|
1
|
|
|
|
|
|
|
package Apache2::Controller::Dispatch; |
|
2
|
|
|
|
|
|
|
|
|
3
|
|
|
|
|
|
|
=head1 NAME |
|
4
|
|
|
|
|
|
|
|
|
5
|
|
|
|
|
|
|
Apache2::Controller::Dispatch - dispatch base class for Apache::Controller |
|
6
|
|
|
|
|
|
|
|
|
7
|
|
|
|
|
|
|
=head1 VERSION |
|
8
|
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
Version 1.001.001 |
|
10
|
|
|
|
|
|
|
|
|
11
|
|
|
|
|
|
|
=cut |
|
12
|
|
|
|
|
|
|
|
|
13
|
2
|
|
|
2
|
|
3163
|
use version; |
|
|
2
|
|
|
|
|
3
|
|
|
|
2
|
|
|
|
|
16
|
|
|
14
|
|
|
|
|
|
|
our $VERSION = version->new('1.001.001'); |
|
15
|
|
|
|
|
|
|
|
|
16
|
|
|
|
|
|
|
=head1 SYNOPSIS |
|
17
|
|
|
|
|
|
|
|
|
18
|
|
|
|
|
|
|
Synopsis examples use L, |
|
19
|
|
|
|
|
|
|
but you may want to check out L |
|
20
|
|
|
|
|
|
|
or write your own. All use the C<< A2C_Dispatch_Map >> directive, |
|
21
|
|
|
|
|
|
|
but the hash structure differs between subclasses. |
|
22
|
|
|
|
|
|
|
|
|
23
|
|
|
|
|
|
|
=head2 EASY WAY |
|
24
|
|
|
|
|
|
|
|
|
25
|
|
|
|
|
|
|
This only works if you have one website on the whole server (under forked mpm) |
|
26
|
|
|
|
|
|
|
because the intepreter only loads the module once and then it won't |
|
27
|
|
|
|
|
|
|
load another dispatch map for other uri's. |
|
28
|
|
|
|
|
|
|
|
|
29
|
|
|
|
|
|
|
# vhost.conf: |
|
30
|
|
|
|
|
|
|
|
|
31
|
|
|
|
|
|
|
SetHandler modperl |
|
32
|
|
|
|
|
|
|
A2C_Dispatch_Map /path/to/yaml/syck/dispatch/hash/file.yaml |
|
33
|
|
|
|
|
|
|
PerlInitHandler Apache2::Controller::Dispatch::Simple |
|
34
|
|
|
|
|
|
|
|
|
35
|
|
|
|
|
|
|
|
|
36
|
|
|
|
|
|
|
=head2 NORMAL WAY |
|
37
|
|
|
|
|
|
|
|
|
38
|
|
|
|
|
|
|
The normal way supports many separate dispatch maps on a server, |
|
39
|
|
|
|
|
|
|
but each application must subclass a dispatch class, even if it |
|
40
|
|
|
|
|
|
|
has no methods. |
|
41
|
|
|
|
|
|
|
|
|
42
|
|
|
|
|
|
|
# vhost.conf: |
|
43
|
|
|
|
|
|
|
PerlLoadModule MyApp::Dispatch; |
|
44
|
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
SetHandler modperl |
|
46
|
|
|
|
|
|
|
A2C_Dispatch_Map /etc/myapp/dispatch.yaml |
|
47
|
|
|
|
|
|
|
PerlInitHandler MyApp::Dispatch |
|
48
|
|
|
|
|
|
|
|
|
49
|
|
|
|
|
|
|
|
|
50
|
|
|
|
|
|
|
# /etc/myapp/dispatch.yaml: |
|
51
|
|
|
|
|
|
|
foo: MyApp::Controller::Foo |
|
52
|
|
|
|
|
|
|
bar: MyApp::Controller::Bar |
|
53
|
|
|
|
|
|
|
'foo/bar': MyApp::Controller::Foo::Bar |
|
54
|
|
|
|
|
|
|
biz: MyApp::C::Biz |
|
55
|
|
|
|
|
|
|
'biz/baz': MyApp::Controller::Biz::Baz |
|
56
|
|
|
|
|
|
|
|
|
57
|
|
|
|
|
|
|
# lib/MyApp/Dispatch.pm: |
|
58
|
|
|
|
|
|
|
package MyApp::Dispatch; |
|
59
|
|
|
|
|
|
|
use base qw( Apache2::Controller::Dispatch::Simple ); |
|
60
|
|
|
|
|
|
|
1; |
|
61
|
|
|
|
|
|
|
|
|
62
|
|
|
|
|
|
|
=head2 HARD WAY |
|
63
|
|
|
|
|
|
|
|
|
64
|
|
|
|
|
|
|
# vhost.conf: |
|
65
|
|
|
|
|
|
|
PerlModule MyApp::Dispatch |
|
66
|
|
|
|
|
|
|
|
|
67
|
|
|
|
|
|
|
|
|
68
|
|
|
|
|
|
|
SetHandler modperl |
|
69
|
|
|
|
|
|
|
PerlInitHandler MyApp::Dispatch |
|
70
|
|
|
|
|
|
|
|
|
71
|
|
|
|
|
|
|
|
|
72
|
|
|
|
|
|
|
# lib/MyApp/Dispatch.pm: |
|
73
|
|
|
|
|
|
|
|
|
74
|
|
|
|
|
|
|
package MyApp::Dispatch; |
|
75
|
|
|
|
|
|
|
|
|
76
|
|
|
|
|
|
|
use strict; |
|
77
|
|
|
|
|
|
|
use warnings FATAL => 'all'; |
|
78
|
|
|
|
|
|
|
|
|
79
|
|
|
|
|
|
|
use base qw( Apache2::Controller::Dispatch::Simple ); |
|
80
|
|
|
|
|
|
|
|
|
81
|
|
|
|
|
|
|
# return a hash reference from dispatch_map() |
|
82
|
|
|
|
|
|
|
sub dispatch_map { return { |
|
83
|
|
|
|
|
|
|
foo => 'MyApp::C::Foo', |
|
84
|
|
|
|
|
|
|
bar => 'MyApp::C::Bar', |
|
85
|
|
|
|
|
|
|
'foo/bar' => 'MyApp::C::Foo::Bar', |
|
86
|
|
|
|
|
|
|
biz => 'MyApp::C::Biz', |
|
87
|
|
|
|
|
|
|
'biz/baz' => 'MyApp::C::Biz::Baz', |
|
88
|
|
|
|
|
|
|
} } |
|
89
|
|
|
|
|
|
|
|
|
90
|
|
|
|
|
|
|
# or use directive A2C_Dispatch_Map to refer to a YAML file. |
|
91
|
|
|
|
|
|
|
|
|
92
|
|
|
|
|
|
|
1; |
|
93
|
|
|
|
|
|
|
|
|
94
|
|
|
|
|
|
|
=head1 DESCRIPTION |
|
95
|
|
|
|
|
|
|
|
|
96
|
|
|
|
|
|
|
C forms the base for the |
|
97
|
|
|
|
|
|
|
PerlInitHandler module to dispatch incoming requests to |
|
98
|
|
|
|
|
|
|
libraries based on their URL. |
|
99
|
|
|
|
|
|
|
|
|
100
|
|
|
|
|
|
|
You don't use this module. You use one of its subclasses |
|
101
|
|
|
|
|
|
|
as a base for your dispatch module. |
|
102
|
|
|
|
|
|
|
|
|
103
|
|
|
|
|
|
|
=head1 WHY A MAP? |
|
104
|
|
|
|
|
|
|
|
|
105
|
|
|
|
|
|
|
Natively, this does not try to figure out the appropriate |
|
106
|
|
|
|
|
|
|
module using any complex magic. Instead, you spell out the |
|
107
|
|
|
|
|
|
|
uris under the handler location and what controller |
|
108
|
|
|
|
|
|
|
modules you want to handle paths under that URL, using a |
|
109
|
|
|
|
|
|
|
directive. (L) |
|
110
|
|
|
|
|
|
|
|
|
111
|
|
|
|
|
|
|
The trouble with automatic controller module detectors is |
|
112
|
|
|
|
|
|
|
that parsing the URI and doing C<< eval "use lib $blah" >> |
|
113
|
|
|
|
|
|
|
up through the URI path is that is computationally expensive. |
|
114
|
|
|
|
|
|
|
|
|
115
|
|
|
|
|
|
|
Maintaining a URI map file is not that difficult and also is |
|
116
|
|
|
|
|
|
|
convenient because you can move libraries around, point different |
|
117
|
|
|
|
|
|
|
URI's to the same controller library, etc. For example to bring |
|
118
|
|
|
|
|
|
|
part of your site off-line and see 'under construction', create |
|
119
|
|
|
|
|
|
|
a controller to print the right message, change all the uri's |
|
120
|
|
|
|
|
|
|
in the map and bump the server. |
|
121
|
|
|
|
|
|
|
|
|
122
|
|
|
|
|
|
|
(Can I trap a signal so it |
|
123
|
|
|
|
|
|
|
clears and reloads map files if Apache2 is HUP'd? That would be cool. |
|
124
|
|
|
|
|
|
|
Or a timeout that would cause children to reload the file.) |
|
125
|
|
|
|
|
|
|
|
|
126
|
|
|
|
|
|
|
Different dispatch types use different structure in the |
|
127
|
|
|
|
|
|
|
map, but it is conceptually the same. The structure is |
|
128
|
|
|
|
|
|
|
loaded into memory and then the uri can be parsed very |
|
129
|
|
|
|
|
|
|
quickly to locate the correct controller. |
|
130
|
|
|
|
|
|
|
|
|
131
|
|
|
|
|
|
|
=head1 SUBCLASSES |
|
132
|
|
|
|
|
|
|
|
|
133
|
|
|
|
|
|
|
Subclasses of this module implement C<< find_controller() >> |
|
134
|
|
|
|
|
|
|
in different ways, usually interpreting the URI from a |
|
135
|
|
|
|
|
|
|
hash reference returned by C<< dispatch_map() >> in your subclass. |
|
136
|
|
|
|
|
|
|
Or, if you provide the directive C<< A2C_Dispatch_Map >> to specify |
|
137
|
|
|
|
|
|
|
a map file, this module will load it with L. |
|
138
|
|
|
|
|
|
|
|
|
139
|
|
|
|
|
|
|
See L and |
|
140
|
|
|
|
|
|
|
L for other |
|
141
|
|
|
|
|
|
|
dispatch possibilities. |
|
142
|
|
|
|
|
|
|
|
|
143
|
|
|
|
|
|
|
Any implementation of find_controller() should throw an |
|
144
|
|
|
|
|
|
|
L with status C<< Apache2::Const::NOT_FOUND >> |
|
145
|
|
|
|
|
|
|
in the |
|
146
|
|
|
|
|
|
|
event that the detected method selected does not appear in the list of |
|
147
|
|
|
|
|
|
|
C<< allowed_methods() >> in the controller module. ex: |
|
148
|
|
|
|
|
|
|
|
|
149
|
|
|
|
|
|
|
a2cx status => Apache2::Const::NOT_FOUND; |
|
150
|
|
|
|
|
|
|
|
|
151
|
|
|
|
|
|
|
See L. This is |
|
152
|
|
|
|
|
|
|
internal stuff mostly, you don't have to implement your own |
|
153
|
|
|
|
|
|
|
type of dispatch mechanism unless you are a nut like me. |
|
154
|
|
|
|
|
|
|
|
|
155
|
|
|
|
|
|
|
Successful run of find_controller() should result in four items of |
|
156
|
|
|
|
|
|
|
data being set in request->pnotes->{a2c}: |
|
157
|
|
|
|
|
|
|
|
|
158
|
|
|
|
|
|
|
=over 4 |
|
159
|
|
|
|
|
|
|
|
|
160
|
|
|
|
|
|
|
=item pnotes->{a2c}{relative_uri} = matching part of uri relative to location |
|
161
|
|
|
|
|
|
|
|
|
162
|
|
|
|
|
|
|
This is the uri relative to the location. For example, |
|
163
|
|
|
|
|
|
|
if the dispatch module is the init handler in a C<< >> |
|
164
|
|
|
|
|
|
|
config block, then for /subdir/foo/bar/biz/zip in this example code, |
|
165
|
|
|
|
|
|
|
relative_uri should be 'foo/bar' because this is the key of %dispatch_map |
|
166
|
|
|
|
|
|
|
that was matched. /subdir/foo/bar is the 'virtual directory.' |
|
167
|
|
|
|
|
|
|
|
|
168
|
|
|
|
|
|
|
If there is no relative uri, for example if the uri requested was /subdir |
|
169
|
|
|
|
|
|
|
and this is the same as the location, then |
|
170
|
|
|
|
|
|
|
C< pnotes->{a2c}{relative_uri} > would be set to |
|
171
|
|
|
|
|
|
|
the empty string. |
|
172
|
|
|
|
|
|
|
|
|
173
|
|
|
|
|
|
|
=item pnotes->{a2c}{controller} = selected package name |
|
174
|
|
|
|
|
|
|
|
|
175
|
|
|
|
|
|
|
This should be the name (string) of an Apache2::Controller subclass selected |
|
176
|
|
|
|
|
|
|
for dispatch. |
|
177
|
|
|
|
|
|
|
|
|
178
|
|
|
|
|
|
|
=item pnotes->{a2c}{method} = method name in controller to process the uri |
|
179
|
|
|
|
|
|
|
|
|
180
|
|
|
|
|
|
|
This is the name of the method of the controller to use for this request. |
|
181
|
|
|
|
|
|
|
|
|
182
|
|
|
|
|
|
|
=item pnotes->{a2c}{path_args} = [ remaining path_info ] |
|
183
|
|
|
|
|
|
|
|
|
184
|
|
|
|
|
|
|
The remaining 'virtual directory' arguments of the uri. |
|
185
|
|
|
|
|
|
|
In the example above for pnotes->{a2c}{relative_uri}, this is [ 'biz', 'zip' ]. |
|
186
|
|
|
|
|
|
|
|
|
187
|
|
|
|
|
|
|
=back |
|
188
|
|
|
|
|
|
|
|
|
189
|
|
|
|
|
|
|
@path_args is the array of remaining elements. For example if your |
|
190
|
|
|
|
|
|
|
dispatch map contains the URI 'foo', and the incoming URI was '/foo/bar/baz', |
|
191
|
|
|
|
|
|
|
then $r->pnotes->{a2c}{path_args} should be ['bar', 'baz'] before returning. |
|
192
|
|
|
|
|
|
|
|
|
193
|
|
|
|
|
|
|
=cut |
|
194
|
|
|
|
|
|
|
|
|
195
|
2
|
|
|
2
|
|
375
|
use strict; |
|
|
2
|
|
|
|
|
3
|
|
|
|
2
|
|
|
|
|
67
|
|
|
196
|
2
|
|
|
2
|
|
11
|
use warnings FATAL => 'all'; |
|
|
2
|
|
|
|
|
8
|
|
|
|
2
|
|
|
|
|
80
|
|
|
197
|
2
|
|
|
2
|
|
17
|
use English '-no_match_vars'; |
|
|
2
|
|
|
|
|
4
|
|
|
|
2
|
|
|
|
|
11
|
|
|
198
|
|
|
|
|
|
|
|
|
199
|
2
|
|
|
|
|
721
|
use base qw( |
|
200
|
|
|
|
|
|
|
Apache2::Controller::NonResponseBase |
|
201
|
|
|
|
|
|
|
Apache2::Controller::Methods |
|
202
|
2
|
|
|
2
|
|
1142
|
); |
|
|
2
|
|
|
|
|
2
|
|
|
203
|
|
|
|
|
|
|
|
|
204
|
|
|
|
|
|
|
use Log::Log4perl qw(:easy); |
|
205
|
|
|
|
|
|
|
use Readonly; |
|
206
|
|
|
|
|
|
|
|
|
207
|
|
|
|
|
|
|
use YAML::Syck; |
|
208
|
|
|
|
|
|
|
|
|
209
|
|
|
|
|
|
|
use Apache2::RequestRec (); |
|
210
|
|
|
|
|
|
|
use Apache2::Connection (); |
|
211
|
|
|
|
|
|
|
use Apache2::RequestUtil (); |
|
212
|
|
|
|
|
|
|
use Apache2::Const -compile => qw( :common :http :methods ); |
|
213
|
|
|
|
|
|
|
|
|
214
|
|
|
|
|
|
|
use Apache2::Controller::X; |
|
215
|
|
|
|
|
|
|
use Apache2::Controller::Const qw( @RANDCHARS $NOT_GOOD_CHARS ); |
|
216
|
|
|
|
|
|
|
use Apache2::Controller::Funk qw( log_bad_request_reason ); |
|
217
|
|
|
|
|
|
|
|
|
218
|
|
|
|
|
|
|
=head1 METHODS |
|
219
|
|
|
|
|
|
|
|
|
220
|
|
|
|
|
|
|
=head2 $handler->process() |
|
221
|
|
|
|
|
|
|
|
|
222
|
|
|
|
|
|
|
process() is the main guts of Apache2::Controller::Dispatch logic. |
|
223
|
|
|
|
|
|
|
It calls $self->find_controller(), which is implemented in another |
|
224
|
|
|
|
|
|
|
base class. (See L.) If that |
|
225
|
|
|
|
|
|
|
works, then it creates an Apache2::Request object from $r, which will |
|
226
|
|
|
|
|
|
|
supposedly parse the query string once for all further handlers that |
|
227
|
|
|
|
|
|
|
create Apache2::Request objects. |
|
228
|
|
|
|
|
|
|
|
|
229
|
|
|
|
|
|
|
=cut |
|
230
|
|
|
|
|
|
|
|
|
231
|
|
|
|
|
|
|
sub process { |
|
232
|
|
|
|
|
|
|
my ($self) = @_; |
|
233
|
|
|
|
|
|
|
|
|
234
|
|
|
|
|
|
|
my $r = $self->{r}; |
|
235
|
|
|
|
|
|
|
my $class = $self->{class}; |
|
236
|
|
|
|
|
|
|
|
|
237
|
|
|
|
|
|
|
my $pnotes = $r->pnotes; |
|
238
|
|
|
|
|
|
|
|
|
239
|
|
|
|
|
|
|
# find the controller module and method to dispatch the URI |
|
240
|
|
|
|
|
|
|
$self->find_controller(); |
|
241
|
|
|
|
|
|
|
my $controller = $self->{controller} = $pnotes->{a2c}{controller}; |
|
242
|
|
|
|
|
|
|
DEBUG "found controller '$controller'"; |
|
243
|
|
|
|
|
|
|
|
|
244
|
|
|
|
|
|
|
# save the dispatch class name in notes in case we have to |
|
245
|
|
|
|
|
|
|
# re-dispatch somewhere along the line if the uri changes |
|
246
|
|
|
|
|
|
|
# (this is done by Apache2::Controller::Auth::OpenID, for instance) |
|
247
|
|
|
|
|
|
|
$pnotes->{a2c}{dispatch_class} = $class; |
|
248
|
|
|
|
|
|
|
|
|
249
|
|
|
|
|
|
|
# set the handler for that class |
|
250
|
|
|
|
|
|
|
# - this has to be the last thing it does in case an exception is thrown |
|
251
|
|
|
|
|
|
|
|
|
252
|
|
|
|
|
|
|
DEBUG "setting PerlResponseHandler '$controller'"; |
|
253
|
|
|
|
|
|
|
$r->set_handlers(PerlResponseHandler => [ "$controller" ]); |
|
254
|
|
|
|
|
|
|
# "" == lame but true, must stringify lib name because |
|
255
|
|
|
|
|
|
|
# the value is some kind of blessed scalar reference or something |
|
256
|
|
|
|
|
|
|
|
|
257
|
|
|
|
|
|
|
DEBUG sub { "Done with process() for uri ".$r->uri }; |
|
258
|
|
|
|
|
|
|
|
|
259
|
|
|
|
|
|
|
return Apache2::Const::OK; |
|
260
|
|
|
|
|
|
|
} |
|
261
|
|
|
|
|
|
|
|
|
262
|
|
|
|
|
|
|
=head2 dispatch_map |
|
263
|
|
|
|
|
|
|
|
|
264
|
|
|
|
|
|
|
The base class method relies on having directive C<< A2C_Dispatch_Map >>. |
|
265
|
|
|
|
|
|
|
This loads a L file at server startup for every instance |
|
266
|
|
|
|
|
|
|
of the directive. This is your best bet if you want to use a file, |
|
267
|
|
|
|
|
|
|
because the file will be loaded only once, instead of every time a |
|
268
|
|
|
|
|
|
|
mod_perl child process spawns. |
|
269
|
|
|
|
|
|
|
|
|
270
|
|
|
|
|
|
|
If you want to return a hash yourself, overload this in a |
|
271
|
|
|
|
|
|
|
dispatch subclass. |
|
272
|
|
|
|
|
|
|
|
|
273
|
|
|
|
|
|
|
=cut |
|
274
|
|
|
|
|
|
|
|
|
275
|
|
|
|
|
|
|
sub dispatch_map { |
|
276
|
|
|
|
|
|
|
my ($self) = @_; |
|
277
|
|
|
|
|
|
|
return $self->get_directive('A2C_Dispatch_Map') |
|
278
|
|
|
|
|
|
|
|| a2cx "No directive A2C_Dispatch_Map"; |
|
279
|
|
|
|
|
|
|
} |
|
280
|
|
|
|
|
|
|
|
|
281
|
|
|
|
|
|
|
=head2 get_dispatch_map |
|
282
|
|
|
|
|
|
|
|
|
283
|
|
|
|
|
|
|
Get the cached C<< \%dispatch_map >> of the dispatch handler object's class. |
|
284
|
|
|
|
|
|
|
Caches references here in parent package space and checks with C<< exists >>. |
|
285
|
|
|
|
|
|
|
|
|
286
|
|
|
|
|
|
|
In your dispatch subclass, you define C<< dispatch_map() >> which |
|
287
|
|
|
|
|
|
|
returns a hash reference of the dispatch map. |
|
288
|
|
|
|
|
|
|
|
|
289
|
|
|
|
|
|
|
=cut |
|
290
|
|
|
|
|
|
|
|
|
291
|
|
|
|
|
|
|
my %dispatch_maps = ( ); |
|
292
|
|
|
|
|
|
|
sub get_dispatch_map { |
|
293
|
|
|
|
|
|
|
my ($self) = @_; |
|
294
|
|
|
|
|
|
|
my $class = $self->{class}; |
|
295
|
|
|
|
|
|
|
return $dispatch_maps{$class} if exists $dispatch_maps{$class}; |
|
296
|
|
|
|
|
|
|
|
|
297
|
|
|
|
|
|
|
my $dispatch_map = $self->dispatch_map(); |
|
298
|
|
|
|
|
|
|
|
|
299
|
|
|
|
|
|
|
a2cx "No dispatch_map() in $class" if !$dispatch_map; |
|
300
|
|
|
|
|
|
|
|
|
301
|
|
|
|
|
|
|
my $ref = ref $dispatch_map; |
|
302
|
|
|
|
|
|
|
a2cx "Bad dispatch_map() in $class" if !defined $ref || $ref ne 'HASH'; |
|
303
|
|
|
|
|
|
|
|
|
304
|
|
|
|
|
|
|
$dispatch_maps{$class} = $dispatch_map; |
|
305
|
|
|
|
|
|
|
|
|
306
|
|
|
|
|
|
|
DEBUG sub{"dispatch_maps:".Dump(\%dispatch_maps)}; |
|
307
|
|
|
|
|
|
|
|
|
308
|
|
|
|
|
|
|
return $dispatch_map; |
|
309
|
|
|
|
|
|
|
} |
|
310
|
|
|
|
|
|
|
|
|
311
|
|
|
|
|
|
|
|
|
312
|
|
|
|
|
|
|
1; |
|
313
|
|
|
|
|
|
|
|
|
314
|
|
|
|
|
|
|
=head1 EXAMPLE |
|
315
|
|
|
|
|
|
|
|
|
316
|
|
|
|
|
|
|
# configuration for : |
|
317
|
|
|
|
|
|
|
# PerlInitHandler MyApp::Dispatch |
|
318
|
|
|
|
|
|
|
|
|
319
|
|
|
|
|
|
|
package MyApp::Dispatch; |
|
320
|
|
|
|
|
|
|
use base qw( |
|
321
|
|
|
|
|
|
|
Apache2::Controller::Dispatch |
|
322
|
|
|
|
|
|
|
Apache2::Controller::Dispatch::Simple |
|
323
|
|
|
|
|
|
|
); |
|
324
|
|
|
|
|
|
|
|
|
325
|
|
|
|
|
|
|
my @LIMIT_HTTP_METHODS = qw( GET ); |
|
326
|
|
|
|
|
|
|
|
|
327
|
|
|
|
|
|
|
sub dispatch_map { { # return a hash reference |
|
328
|
|
|
|
|
|
|
foo => 'MyApp::C::Foo', |
|
329
|
|
|
|
|
|
|
bar => 'MyApp::C::Bar', |
|
330
|
|
|
|
|
|
|
biz => 'MyApp::C::Biz', |
|
331
|
|
|
|
|
|
|
} } |
|
332
|
|
|
|
|
|
|
|
|
333
|
|
|
|
|
|
|
1; |
|
334
|
|
|
|
|
|
|
|
|
335
|
|
|
|
|
|
|
=head1 SEE ALSO |
|
336
|
|
|
|
|
|
|
|
|
337
|
|
|
|
|
|
|
L |
|
338
|
|
|
|
|
|
|
|
|
339
|
|
|
|
|
|
|
L |
|
340
|
|
|
|
|
|
|
|
|
341
|
|
|
|
|
|
|
L |
|
342
|
|
|
|
|
|
|
|
|
343
|
|
|
|
|
|
|
=head1 AUTHOR |
|
344
|
|
|
|
|
|
|
|
|
345
|
|
|
|
|
|
|
Mark Hedges, C<< >> |
|
346
|
|
|
|
|
|
|
|
|
347
|
|
|
|
|
|
|
=head1 COPYRIGHT & LICENSE |
|
348
|
|
|
|
|
|
|
|
|
349
|
|
|
|
|
|
|
Copyright 2008-2010 Mark Hedges, all rights reserved. |
|
350
|
|
|
|
|
|
|
|
|
351
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify it |
|
352
|
|
|
|
|
|
|
under the same terms as Perl itself. |
|
353
|
|
|
|
|
|
|
|
|
354
|
|
|
|
|
|
|
This software is provided as-is, with no warranty |
|
355
|
|
|
|
|
|
|
and no guarantee of fitness |
|
356
|
|
|
|
|
|
|
for any particular purpose. |
|
357
|
|
|
|
|
|
|
|
|
358
|
|
|
|
|
|
|
=cut |
|
359
|
|
|
|
|
|
|
|