new action method "list_extra_info" for better debugging, and did this for the new...
[catagits/Catalyst-Runtime.git] / lib / Catalyst / Action.pm
1 package Catalyst::Action;
2
3 =head1 NAME
4
5 Catalyst::Action - Catalyst Action
6
7 =head1 SYNOPSIS
8
9     <form action="[%c.uri_for(c.action)%]">
10
11     $c->forward( $action->private_path );
12
13 =head1 DESCRIPTION
14
15 This class represents a Catalyst Action. You can access the object for the
16 currently dispatched action via $c->action. See the L<Catalyst::Dispatcher>
17 for more information on how actions are dispatched. Actions are defined in
18 L<Catalyst::Controller> subclasses.
19
20 =cut
21
22 use Moose;
23 use Scalar::Util 'looks_like_number';
24 with 'MooseX::Emulate::Class::Accessor::Fast';
25 use namespace::clean -except => 'meta';
26
27 has class => (is => 'rw');
28 has namespace => (is => 'rw');
29 has 'reverse' => (is => 'rw');
30 has attributes => (is => 'rw');
31 has name => (is => 'rw');
32 has code => (is => 'rw');
33 has private_path => (
34   reader => 'private_path',
35   isa => 'Str',
36   lazy => 1,
37   required => 1,
38   default => sub { '/'.shift->reverse },
39 );
40
41 use overload (
42
43     # Stringify to reverse for debug output etc.
44     q{""} => sub { shift->{reverse} },
45
46     # Codulate to execute to invoke the encapsulated action coderef
47     '&{}' => sub { my $self = shift; sub { $self->execute(@_); }; },
48
49     # Make general $stuff still work
50     fallback => 1,
51
52 );
53
54
55
56 no warnings 'recursion';
57
58 sub dispatch {    # Execute ourselves against a context
59     my ( $self, $c ) = @_;
60     return $c->execute( $self->class, $self );
61 }
62
63 sub execute {
64   my $self = shift;
65   $self->code->(@_);
66 }
67
68 sub match {
69     my ( $self, $c ) = @_;
70     #would it be unreasonable to store the number of arguments
71     #the action has as its own attribute?
72     #it would basically eliminate the code below.  ehhh. small fish
73     return 1 unless exists $self->attributes->{Args};
74     my $args = $self->attributes->{Args}[0];
75     return 1 unless defined($args) && length($args);
76     return scalar( @{ $c->req->args } ) == $args;
77 }
78
79 sub match_captures { 1 }
80
81 sub compare {
82     my ($a1, $a2) = @_;
83
84     my ($a1_args) = @{ $a1->attributes->{Args} || [] };
85     my ($a2_args) = @{ $a2->attributes->{Args} || [] };
86
87     $_ = looks_like_number($_) ? $_ : ~0
88         for $a1_args, $a2_args;
89
90     return $a1_args <=> $a2_args;
91 }
92
93 sub number_of_args {
94     my ( $self ) = @_;
95     return 0 unless exists $self->attributes->{Args};
96     return $self->attributes->{Args}[0];
97 }
98
99 sub number_of_captures {
100     my ( $self ) = @_;
101
102     return 0 unless exists $self->attributes->{CaptureArgs};
103     return $self->attributes->{CaptureArgs}[0] || 0;
104 }
105
106 sub list_extra_info { } 
107
108 __PACKAGE__->meta->make_immutable;
109
110 1;
111
112 __END__
113
114 =head1 METHODS
115
116 =head2 attributes
117
118 The sub attributes that are set for this action, like Local, Path, Private
119 and so on. This determines how the action is dispatched to.
120
121 =head2 class
122
123 Returns the name of the component where this action is defined.
124 Derived by calling the L<Catalyst::Component/catalyst_component_name|catalyst_component_name>
125 method on each component.
126
127 =head2 code
128
129 Returns a code reference to this action.
130
131 =head2 dispatch( $c )
132
133 Dispatch this action against a context.
134
135 =head2 execute( $controller, $c, @args )
136
137 Execute this action's coderef against a given controller with a given
138 context and arguments
139
140 =head2 match( $c )
141
142 Check Args attribute, and makes sure number of args matches the setting.
143 Always returns true if Args is omitted.
144
145 =head2 match_captures ($c, $captures)
146
147 Can be implemented by action class and action role authors. If the method
148 exists, then it will be called with the request context and an array reference
149 of the captures for this action.
150
151 Returning true from this method causes the chain match to continue, returning
152 makes the chain not match (and alternate, less preferred chains will be attempted).
153
154
155 =head2 compare
156
157 Compares 2 actions based on the value of the C<Args> attribute, with no C<Args>
158 having the highest precedence.
159
160 =head2 namespace
161
162 Returns the private namespace this action lives in.
163
164 =head2 reverse
165
166 Returns the private path for this action.
167
168 =head2 private_path
169
170 Returns absolute private path for this action. Unlike C<reverse>, the
171 C<private_path> of an action is always suitable for passing to C<forward>.
172
173 =head2 name
174
175 Returns the sub name of this action.
176
177 =head2 number_of_args
178
179 Returns the number of args this action expects. This is 0 if the action doesn't take any arguments and undef if it will take any number of arguments.
180
181 =head2 number_of_captures
182
183 Returns the number of captures this action expects for L<Chained|Catalyst::DispatchType::Chained> actions.
184
185 =head2 list_extra_info
186
187 An array of values useful to improve debugging
188
189 =head2 meta
190
191 Provided by Moose.
192
193 =head1 AUTHORS
194
195 Catalyst Contributors, see Catalyst.pm
196
197 =head1 COPYRIGHT
198
199 This library is free software. You can redistribute it and/or modify it under
200 the same terms as Perl itself.
201
202 =cut