deprecate Catalyst::Base (left for compability reasons)
[catagits/Catalyst-Runtime.git] / lib / Catalyst / Controller.pm
1 package Catalyst::Controller;
2
3 use strict;
4 use base qw/Catalyst::Component Catalyst::AttrContainer Class::Accessor::Fast/;
5
6 use Catalyst::Exception;
7 use Catalyst::Utils;
8 use Class::Inspector;
9 use NEXT;
10
11 =head1 NAME
12
13 Catalyst::Controller - Catalyst Controller base class
14
15 =head1 SYNOPSIS
16
17   package MyApp::Controller::Search
18   use base qw/Catalyst::Controller;
19
20   sub foo : Local { 
21         my ($self,$c,@args) = @_;
22         ... 
23   } # Dispatches to /search/foo
24
25 =head1 DESCRIPTION
26
27 Controllers are where the actions in the Catalyst framework reside. each
28 action is represented by a function with an attribute to identify what kind
29 of action it is. See the L<Catalyst::Dispatcher> for more info about how 
30 Catalyst dispatches to actions.
31
32 =cut
33
34 __PACKAGE__->mk_classdata($_) for qw/_dispatch_steps _action_class/;
35
36 __PACKAGE__->_dispatch_steps( [qw/_BEGIN _AUTO _ACTION/] );
37 __PACKAGE__->_action_class('Catalyst::Action');
38
39 __PACKAGE__->mk_accessors( qw/_application/ );
40
41 ### _app as alias
42 *_app = *_application;
43
44 sub _DISPATCH : Private {
45     my ( $self, $c ) = @_;
46
47     foreach my $disp ( @{ $self->_dispatch_steps } ) {
48         last unless $c->forward($disp);
49     }
50
51     $c->forward('_END');
52 }
53
54 sub _BEGIN : Private {
55     my ( $self, $c ) = @_;
56     my $begin = ( $c->get_actions( 'begin', $c->namespace ) )[-1];
57     return 1 unless $begin;
58     $begin->dispatch( $c );
59     return !@{ $c->error };
60 }
61
62 sub _AUTO : Private {
63     my ( $self, $c ) = @_;
64     my @auto = $c->get_actions( 'auto', $c->namespace );
65     foreach my $auto (@auto) {
66         $auto->dispatch( $c );
67         return 0 unless $c->state;
68     }
69     return 1;
70 }
71
72 sub _ACTION : Private {
73     my ( $self, $c ) = @_;
74     if (   ref $c->action
75         && $c->action->can('execute')
76         && $c->req->action )
77     {
78         $c->action->dispatch( $c );
79     }
80     return !@{ $c->error };
81 }
82
83 sub _END : Private {
84     my ( $self, $c ) = @_;
85     my $end = ( $c->get_actions( 'end', $c->namespace ) )[-1];
86     return 1 unless $end;
87     $end->dispatch( $c );
88     return !@{ $c->error };
89 }
90
91 sub new {
92     my $self = shift;
93     my $app = $_[0];
94     my $new = $self->NEXT::new(@_);
95     $new->_application( $app );
96     return $new;
97 }
98
99
100 sub action_for {
101     my ( $self, $name ) = @_;
102     my $app = ($self->isa('Catalyst') ? $self : $self->_application);
103     return $app->dispatcher->get_action($name, $self->action_namespace);
104 }
105
106 sub action_namespace {
107     my ( $self, $c ) = @_;
108     unless ( $c ) {
109         $c = ($self->isa('Catalyst') ? $self : $self->_application);
110     }
111     my $hash = (ref $self ? $self : $self->config); # hate app-is-class
112     return $hash->{namespace} if exists $hash->{namespace};
113     return Catalyst::Utils::class2prefix( ref($self) || $self,
114         $c->config->{case_sensitive} )
115       || '';
116 }
117
118 sub path_prefix {
119     my ( $self, $c ) = @_;
120     unless ( $c ) {
121         $c = ($self->isa('Catalyst') ? $self : $self->_application);
122     }
123     my $hash = (ref $self ? $self : $self->config); # hate app-is-class
124     return $hash->{path} if exists $hash->{path};
125     return shift->action_namespace(@_);
126 }
127
128
129 sub register_actions {
130     my ( $self, $c ) = @_;
131     my $class = ref $self || $self;
132     my $namespace = $self->action_namespace($c);
133     my %methods;
134     $methods{ $self->can($_) } = $_
135       for @{ Class::Inspector->methods($class) || [] };
136
137     # Advanced inheritance support for plugins and the like
138     my @action_cache;
139     {
140         no strict 'refs';
141         for my $isa ( @{"$class\::ISA"}, $class ) {
142             push @action_cache, @{ $isa->_action_cache }
143               if $isa->can('_action_cache');
144         }
145     }
146
147     foreach my $cache (@action_cache) {
148         my $code   = $cache->[0];
149         my $method = delete $methods{$code}; # avoid dupe registers
150         next unless $method;
151         my $attrs = $self->_parse_attrs( $c, $method, @{ $cache->[1] } );
152         if ( $attrs->{Private} && ( keys %$attrs > 1 ) ) {
153             $c->log->debug( 'Bad action definition "'
154                   . join( ' ', @{ $cache->[1] } )
155                   . qq/" for "$class->$method"/ )
156               if $c->debug;
157             next;
158         }
159         my $reverse = $namespace ? "$namespace/$method" : $method;
160         my $action = $self->create_action(
161             name       => $method,
162             code       => $code,
163             reverse    => $reverse,
164             namespace  => $namespace,
165             class      => $class,
166             attributes => $attrs,
167         );
168
169         $c->dispatcher->register( $c, $action );
170     }
171 }
172
173 sub create_action {
174     my $self = shift;
175     my %args = @_;
176
177     my $class = (exists $args{attributes}{ActionClass}
178                     ? $args{attributes}{ActionClass}[0]
179                     : $self->_action_class);
180
181     unless ( Class::Inspector->loaded($class) ) {
182         require Class::Inspector->filename($class);
183     }
184     
185     return $class->new( \%args );
186 }
187
188 sub _parse_attrs {
189     my ( $self, $c, $name, @attrs ) = @_;
190
191     my %raw_attributes;
192
193     foreach my $attr (@attrs) {
194
195         # Parse out :Foo(bar) into Foo => bar etc (and arrayify)
196
197         if ( my ( $key, $value ) = ( $attr =~ /^(.*?)(?:\(\s*(.+?)\s*\))?$/ ) )
198         {
199
200             if ( defined $value ) {
201                 ( $value =~ s/^'(.*)'$/$1/ ) || ( $value =~ s/^"(.*)"/$1/ );
202             }
203             push( @{ $raw_attributes{$key} }, $value );
204         }
205     }
206
207     my $hash = (ref $self ? $self : $self->config); # hate app-is-class
208
209     if (exists $hash->{actions} || exists $hash->{action}) {
210       my $a = $hash->{actions} || $hash->{action};
211       %raw_attributes = ((exists $a->{'*'} ? %{$a->{'*'}} : ()),
212                          %raw_attributes,
213                          (exists $a->{$name} ? %{$a->{$name}} : ()));
214     }
215
216     my %final_attributes;
217
218     foreach my $key (keys %raw_attributes) {
219
220         my $raw = $raw_attributes{$key};
221
222         foreach my $value (ref($raw) eq 'ARRAY' ? @$raw : $raw) {
223
224             my $meth = "_parse_${key}_attr";
225             if ( $self->can($meth) ) {
226                 ( $key, $value ) = $self->$meth( $c, $name, $value );
227             }
228             push( @{ $final_attributes{$key} }, $value );
229         }
230     }
231
232     return \%final_attributes;
233 }
234
235 sub _parse_Global_attr {
236     my ( $self, $c, $name, $value ) = @_;
237     return $self->_parse_Path_attr( $c, $name, "/$name" );
238 }
239
240 sub _parse_Absolute_attr { shift->_parse_Global_attr(@_); }
241
242 sub _parse_Local_attr {
243     my ( $self, $c, $name, $value ) = @_;
244     return $self->_parse_Path_attr( $c, $name, $name );
245 }
246
247 sub _parse_Relative_attr { shift->_parse_Local_attr(@_); }
248
249 sub _parse_Path_attr {
250     my ( $self, $c, $name, $value ) = @_;
251     $value ||= '';
252     if ( $value =~ m!^/! ) {
253         return ( 'Path', $value );
254     }
255     elsif ( length $value ) {
256         return ( 'Path', join( '/', $self->path_prefix($c), $value ) );
257     }
258     else {
259         return ( 'Path', $self->path_prefix($c) );
260     }
261 }
262
263 sub _parse_Regex_attr {
264     my ( $self, $c, $name, $value ) = @_;
265     return ( 'Regex', $value );
266 }
267
268 sub _parse_Regexp_attr { shift->_parse_Regex_attr(@_); }
269
270 sub _parse_LocalRegex_attr {
271     my ( $self, $c, $name, $value ) = @_;
272     unless ( $value =~ s/^\^// ) { $value = "(?:.*?)$value"; }
273     return ( 'Regex', '^' . $self->path_prefix($c) . "/${value}" );
274 }
275
276 sub _parse_LocalRegexp_attr { shift->_parse_LocalRegex_attr(@_); }
277
278 sub _parse_ActionClass_attr {
279     my ( $self, $c, $name, $value ) = @_;
280     unless ( $value =~ s/^\+// ) {
281       $value = join('::', $self->_action_class, $value );
282     }
283     return ( 'ActionClass', $value );
284 }
285
286
287
288 1;
289
290 __END__
291
292 =head1 CONFIGURATION
293
294 As any other L<Catalyst::Component>, controllers have a config 
295 hash, accessable through $self->config from the controller actions.
296 Some settings are in use by the Catalyst framework:
297
298 =head2 namespace
299
300 This spesifies the internal namespace the controller should be bound to. By default
301 the controller is bound to the uri version of the controller name. For instance
302 controller 'MyApp::Controller::Foo::Bar' will be bound to 'foo/bar'. The default Root 
303 controller is an example of setting namespace to ''. 
304
305 =head2 prefix 
306
307 Sets 'path_prefix', as described below.
308
309 =head1 METHODS
310
311 =head2 $class->new($app, @args)
312
313 Proxies through to NEXT::new and stashes the application instance as
314 $self->_application.
315
316 =head2 $self->action_for('name')
317
318 Returns the Catalyst::Action object (if any) for a given method name in
319 this component.
320
321 =head2 $self->register_actions($c)
322
323 Finds all applicable actions for this component, creates Catalyst::Action
324 objects (using $self->create_action) for them and registers them with
325 $c->dispatcher.
326
327 =head2 $self->action_namespace($c)
328
329 Returns the private namespace for actions in this component. Defaults to a value
330 from the controller name (for e.g. MyApp::Controller::Foo::Bar becomes
331 "foo/bar") or can be overriden from the "namespace" config key.
332
333
334 =head2 $self->path_prefix($c)
335
336 Returns the default path prefix for :Local, :LocalRegex and relative :Path
337 actions in this component. Defaults to the action_namespace or can be
338 overriden from the "path" config key.
339
340 =head2 $self->create_action(%args)
341
342 Called with a hash of data to be use for construction of a new Catalyst::Action
343 (or appropriate sub/alternative class) object.
344
345 Primarily designed for the use of register_actions.
346
347 =head2 $self->_application  
348
349 =head2 $self->_app
350
351 Returns the application instance stored by C<new()>
352
353 =head1 AUTHOR
354
355 Sebastian Riedel, C<sri@oook.de>
356 Marcus Ramberg C<mramberg@cpan.org>
357
358 =head1 COPYRIGHT
359
360 This program is free software, you can redistribute it and/or modify it under
361 the same terms as Perl itself.
362
363 =cut