X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?p=catagits%2FCatalyst-Runtime.git;a=blobdiff_plain;f=lib%2FCatalyst%2FDispatcher.pm;h=a2a4e1c353fe03a824eeb068f489ab37ff56c04a;hp=12040b23654b7e8c798e8125bee52e9311b3b9ec;hb=dd4530ecdc4684838d9c0e9dc00adebb6100b022;hpb=0ca510f0aa1cabe138d81897d38111d7b772449c diff --git a/lib/Catalyst/Dispatcher.pm b/lib/Catalyst/Dispatcher.pm index 12040b2..a2a4e1c 100644 --- a/lib/Catalyst/Dispatcher.pm +++ b/lib/Catalyst/Dispatcher.pm @@ -16,6 +16,7 @@ use Tree::Simple; use Tree::Simple::Visitor::FindByPath; use Class::Load qw(load_class try_load_class); use Encode 2.21 'decode_utf8'; +use Ref::Util qw(is_plain_arrayref is_plain_coderef); use namespace::clean -except => 'meta'; @@ -135,11 +136,11 @@ sub _command2action { my (@args, @captures); - if ( ref( $extra_params[-2] ) eq 'ARRAY' ) { + if ( is_plain_arrayref($extra_params[-2]) ) { @captures = @{ splice @extra_params, -2, 1 }; } - if ( ref( $extra_params[-1] ) eq 'ARRAY' ) { + if ( is_plain_arrayref($extra_params[-1]) ) { @args = @{ pop @extra_params } } else { # this is a copy, it may take some abuse from @@ -239,7 +240,7 @@ Documented in L sub forward { my $self = shift; no warnings 'recursion'; - $self->_do_forward(forward => @_); + return $self->_do_forward(forward => @_); } sub _do_forward { @@ -261,6 +262,12 @@ sub _do_forward { no warnings 'recursion'; $action->dispatch( $c ); + #If there is an error, all bets off regarding state. Documentation + #Specifies that when you forward, if there's an error you must expect + #state to be 0. + if( @{ $c->error }) { + $c->state(0); + } return $c->state; } @@ -273,6 +280,7 @@ Documented in L sub detach { my ( $self, $c, $command, @args ) = @_; $self->_do_forward(detach => $c, $command, @args ) if $command; + $c->state(0); # Needed in order to skip any auto functions Catalyst::Exception::Detach->throw; } @@ -400,9 +408,14 @@ sub prepare_action { if ( $c->debug && @args ); } -=head2 $self->get_action( $action, $namespace ) +=head2 $self->get_action( $action_name, $namespace ) + +returns a named action from a given namespace. C<$action_name> +may be a relative path on that C<$namespace> such as -returns a named action from a given namespace. + $self->get_action('../bar', 'foo/baz'); + +In which case we look for the action at 'foo/bar'. =cut @@ -412,17 +425,22 @@ sub get_action { $namespace = join( "/", grep { length } split '/', ( defined $namespace ? $namespace : "" ) ); - return $self->_action_hash->{"${namespace}/${name}"}; + return $self->get_action_by_path("${namespace}/${name}"); } =head2 $self->get_action_by_path( $path ); Returns the named action by its full private path. +This method performs some normalization on C<$path> so that if +it includes '..' it will do the right thing (for example if +C<$path> is '/foo/../bar' that is normalized to '/bar'. + =cut sub get_action_by_path { my ( $self, $path ) = @_; + $path =~s/[^\/]+\/\.\.\/// while $path=~m/[^\/]+\/\.\.\//; $path =~ s/^\///; $path = "/$path" unless $path =~ /\//; $self->_action_hash->{$path}; @@ -613,7 +631,8 @@ sub setup_actions { $self->_load_dispatch_types( @{ $self->preload_dispatch_types } ); @{ $self->_registered_dispatch_types }{@classes} = (1) x @classes; - foreach my $comp ( values %{ $c->components } ) { + foreach my $comp ( map @{$_}{sort keys %$_}, $c->components ) { + $comp = $comp->() if is_plain_coderef($comp); $comp->register_actions($c) if $comp->can('register_actions'); }