sub dispatch {
my ( $self, $c ) = @_;
-
if ( $c->action ) {
- $c->forward( join( '/', '', $c->namespace, '_DISPATCH' ) );
+ $c->forward( join( '/', '', $c->action->namespace, '_DISPATCH' ) );
}
else {
return 0;
}
- my $arguments = ( ref( $_[-1] ) eq 'ARRAY' ) ? pop(@_) : $c->req->args;
+ my $local_args = 0;
+ my $arguments = $c->req->args;
+ if ( ref( $_[-1] ) eq 'ARRAY' ) {
+ $arguments = pop(@_);
+ $local_args = 1;
+ }
my $result;
my $command_copy = $command;
unless ( $command_copy =~ s/^\/// ) {
- my $namespace = $c->namespace;
+ my $namespace = $c->stack->[-1]->namespace;
$command_copy = "${namespace}/${command}";
}
my $tail = $2;
$result = $c->get_action( $tail, $1 );
if ($result) {
- $command = $tail;
- push( @{$arguments}, @extra_args );
+ $local_args = 1;
+ $command = $tail;
+ unshift( @{$arguments}, @extra_args );
last DESCEND;
}
unshift( @extra_args, $tail );
}
}
}
- else {
- $result = $command;
- }
unless ($result) {
- my $comp;
+ my $class = ref($command) || ref( $c->component($command) );
+ my $method = shift || 'process';
- unless ( $comp = $c->component($command) ) {
+ unless ($class) {
my $error =
qq/Couldn't forward to command "$command". Invalid action or component./;
$c->error($error);
return 0;
}
- my $class = ref $comp;
- my $method = shift || 'process';
-
if ( my $code = $class->can($method) ) {
my $action = $self->method_action_class->new(
{
code => $code,
reverse => "$class->$method",
class => $class,
- namespace => $class,
+ namespace => Catalyst::Utils::class2prefix(
+ $class, $c->config->{case_sensitive}
+ ),
}
);
$result = $action;
}
- local $c->request->{arguments} = [ @{$arguments} ];
-
- $result->execute($c);
+ if ($local_args) {
+ local $c->request->{arguments} = [ @{$arguments} ];
+ $result->execute($c);
+ }
+ else { $result->execute($c) }
return $c->state;
}
}
# If not, move the last part path to args
-
unshift @args, pop @path;
}
+ $c->log->debug( 'Path is "' . $c->req->match . '"' )
+ if ( $c->debug && $c->req->match );
+
$c->log->debug( 'Arguments are "' . join( '/', @args ) . '"' )
if ( $c->debug && @args );
}
-=item $self->get_action( $c, $action, $namespace )
+=item $self->get_action( $action, $namespace )
=cut
sub get_action {
- my ( $self, $c, $name, $namespace ) = @_;
+ my ( $self, $name, $namespace ) = @_;
return unless $name;
$namespace ||= '';
$namespace = '' if $namespace eq '/';
return unless @match;
- if ( my $action = $match[-1]->get_action( $c, $name ) ) {
+ if ( my $action = $match[-1]->get_action($name) ) {
return $action if $action->namespace eq $namespace;
}
}
my @match = $self->get_containers($namespace);
- return map { $_->get_action( $c, $action ) } @match;
+ return map { $_->get_action($action) } @match;
}
=item $self->get_containers( $namespace )
# should catch any failures - or short-circuit this if this *is* a
# bug in the visitor and gets fixed.
- my $extra = $path[ ( scalar @match ) - 1 ];
- last unless $extra;
- $visitor->setSearchPath($extra);
- $match[-1]->accept($visitor);
- push( @match, $visitor->getResult ) if defined $visitor->getResult;
+ if ( my $extra = $path[ ( scalar @match ) - 1 ] ) {
+ $visitor->setSearchPath($extra);
+ $match[-1]->accept($visitor);
+ push( @match, $visitor->getResult ) if defined $visitor->getResult;
+ }
}
return map { $_->getNodeValue } @match;
}
+=item $self->register( $c, $action )
+
+=cut
+
sub register {
my ( $self, $c, $action ) = @_;
+ my $registered = $self->registered_dispatch_types;
+
+ my $priv = 0;
+ foreach my $key ( keys %{ $action->attributes } ) {
+ $priv++ if $key eq 'Private';
+ my $class = "Catalyst::DispatchType::$key";
+ unless ( $registered->{$class} ) {
+ eval "require $class";
+ push( @{ $self->dispatch_types }, $class->new ) unless $@;
+ $registered->{$class} = 1;
+ }
+ }
+
+ # Pass the action to our dispatch types so they can register it if reqd.
+ my $reg = 0;
+ foreach my $type ( @{ $self->dispatch_types } ) {
+ $reg++ if $type->register( $c, $action );
+ }
+
+ return unless $reg + $priv;
+
my $namespace = $action->namespace;
my $parent = $self->tree;
my $visitor = Tree::Simple::Visitor::FindByPath->new;
# Set the method value
$parent->getNodeValue->actions->{ $action->name } = $action;
-
- my $registered = $self->registered_dispatch_types;
-
- foreach my $key ( keys %{ $action->attributes } ) {
- my $class = "Catalyst::DispatchType::$key";
- unless ( $registered->{$class} ) {
- eval "require $class";
- push( @{ $self->dispatch_types }, $class->new ) unless $@;
- $registered->{$class} = 1;
- }
- }
-
- # Pass the action to our dispatch types so they can register it if reqd.
- foreach my $type ( @{ $self->dispatch_types } ) {
- $type->register( $c, $action );
- }
}
=item $self->setup_actions( $class, $component )
Catalyst::ActionContainer->new( { part => '/', actions => {} } );
$self->tree( Tree::Simple->new( $container, Tree::Simple->ROOT ) );
- $c->register_actions($c);
-
foreach my $comp ( values %{ $c->components } ) {
$comp->register_actions($c) if $comp->can('register_actions');
}
return unless $c->debug;
- my $privates = Text::SimpleTable->new( [ 36, 'Private' ], [ 37, 'Class' ] );
+ my $privates = Text::SimpleTable->new(
+ [ 20, 'Private' ],
+ [ 38, 'Class' ],
+ [ 12, 'Method' ]
+ );
my $has_private = 0;
my $walker = sub {
next
if ( ( $action =~ /^_.*/ )
&& ( !$c->config->{show_internal_actions} ) );
- $privates->row( "$prefix$action", $action_obj->class );
+ $privates->row( "$prefix$action", $action_obj->class, $action );
$has_private = 1;
}