X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?p=catagits%2FCatalyst-Runtime.git;a=blobdiff_plain;f=lib%2FCatalyst%2FController.pm;h=fb6df0dfd170c3b1e51729dee95a4f186231c2cf;hp=ca94a3b3bd394b435d91fb4e6c94657987ebe49e;hb=aea897b2495d079f3849eb30cd3dc7c89b31f742;hpb=6904879203b100d3cf3e8f5bf78e2d164476e654 diff --git a/lib/Catalyst/Controller.pm b/lib/Catalyst/Controller.pm index ca94a3b..fb6df0d 100644 --- a/lib/Catalyst/Controller.pm +++ b/lib/Catalyst/Controller.pm @@ -13,28 +13,36 @@ use Catalyst::Utils; with 'Catalyst::Component::ApplicationAttribute'; -has path_prefix => - ( - is => 'rw', - isa => 'Str', - init_arg => 'path', - predicate => 'has_path_prefix', - ); - -has action_namespace => - ( - is => 'rw', - isa => 'Str', - init_arg => 'namespace', - predicate => 'has_action_namespace', - ); - -has actions => - ( - accessor => '_controller_actions', - isa => 'HashRef', - init_arg => undef, - ); +has path_prefix => ( + is => 'rw', + isa => 'Str', + init_arg => 'path', + predicate => 'has_path_prefix', +); + +has action_namespace => ( + is => 'rw', + isa => 'Str', + init_arg => 'namespace', + predicate => 'has_action_namespace', +); + +has actions => ( + accessor => '_controller_actions', + isa => 'HashRef', + init_arg => undef, +); + +has action_args => (is => 'ro'); + +# ->config(actions => { '*' => ... +has _all_actions_attributes => ( + is => 'ro', + isa => 'HashRef', + init_arg => undef, + lazy => 1, + builder => '_build__all_actions_attributes', +); sub BUILD { my ($self, $args) = @_; @@ -42,9 +50,15 @@ sub BUILD { my $actions = delete $args->{actions} || {}; my $attr_value = $self->merge_config_hashes($actions, $action); $self->_controller_actions($attr_value); -} + # trigger lazy builder + $self->_all_actions_attributes; +} +sub _build__all_actions_attributes { + my ($self) = @_; + delete $self->_controller_actions->{'*'} || {}; +} =head1 NAME @@ -195,7 +209,7 @@ sub get_action_methods { $meta->find_method_by_name($_) || confess( sprintf 'Action "%s" is not available from controller %s', $_, ref $self ) - } keys %{ $self->_controller_actions } + } keys %{ $self->_controller_actions } ) if ( ref $self ); return uniq @methods; } @@ -264,7 +278,11 @@ sub create_action { my %args = @_; my $class = $self->action_class(%args); - my $action_args = $self->config->{action_args}; + my $action_args = ( + ref($self) + ? $self->action_args + : $self->config->{action_args} + ); my %extra_args = ( %{ $action_args->{'*'} || {} }, @@ -293,18 +311,32 @@ sub _parse_attrs { } } - my $actions; + my ($actions_config, $all_actions_config); if( ref($self) ) { - $actions = $self->_controller_actions; + $actions_config = $self->_controller_actions; + # No, you're not getting actions => { '*' => ... } with actions in MyApp. + $all_actions_config = $self->_all_actions_attributes; } else { my $cfg = $self->config; - $actions = $self->merge_config_hashes($cfg->{actions}, $cfg->{action}); + $actions_config = $self->merge_config_hashes($cfg->{actions}, $cfg->{action}); + $all_actions_config = {}; } - %raw_attributes = ((exists $actions->{'*'} ? %{$actions->{'*'}} : ()), - %raw_attributes, - (exists $actions->{$name} ? %{$actions->{$name}} : ())); + %raw_attributes = ( + %raw_attributes, + # Note we deep copy array refs here to stop crapping on config + # when attributes are parsed. RT#65463 + exists $actions_config->{$name} ? map { ref($_) eq 'ARRAY' ? [ @$_ ] : $_ } %{ $actions_config->{$name } } : (), + ); + # Private actions with additional attributes will raise a warning and then + # be ignored. Adding '*' arguments to the default _DISPATCH / etc. methods, + # which are Private, will prevent those from being registered. They should + # probably be turned into :Actions instead, or we might want to otherwise + # disambiguate between those built-in internal actions and user-level + # Private ones. + %raw_attributes = (%{ $all_actions_config }, %raw_attributes) + unless $raw_attributes{Private}; my %final_attributes; @@ -453,7 +485,7 @@ Sets 'path_prefix', as described below. Allows you to set the attributes that the dispatcher creates actions out of. This allows you to do 'rails style routes', or override some of the -attribute defintions of actions composed from Roles. +attribute definitions of actions composed from Roles. You can set arguments globally (for all actions of the controller) and specifically (for a single action).