Add types::structured, as we need a calling convention.. Add namespace::autoclean...
[catagits/CatalystX-DynamicComponent.git] / lib / CatalystX / ModelToControllerReflector.pm
CommitLineData
59fc9d16 1package CatalystX::ModelToControllerReflector;
2use Moose::Role;
3use namespace::clean -except => 'meta';
4
6a2f1e96 5with 'CatalystX::DynamicComponent'
6 => { alias => { _setup_dynamic_component => '_setup_dynamic_controller' } };
59fc9d16 7
cbc455a6 8requires 'setup_components';
59fc9d16 9
cbc455a6 10after 'setup_components' => sub { shift->_setup_dynamic_controllers(@_); };
59fc9d16 11
12sub _setup_dynamic_controllers {
13 my ($app) = @_;
14 my @model_names = grep { /::Model::/ } keys %{ $app->components };
15
16 foreach my $model_name (@model_names) {
6a2f1e96 17 $app->_reflect_model_to_controller( $model_name, $app->components->{$model_name} );
59fc9d16 18 }
19}
20
6a2f1e96 21sub _reflect_model_to_controller {
22 my ( $app, $model_name, $model ) = @_;
23
77e54b00 24 my $class = blessed($app) || $app;
25
6a2f1e96 26 my $controller_name = $model_name;
27 $controller_name =~ s/::Model::/::Controller::/;
28
77e54b00 29 my $suffix = $model_name;
30 $suffix =~ s/^.*::Model:://;
31
32 my $controller = $app->_setup_dynamic_controller( $controller_name, {}, sub {
33 shift->next::method(@_); # Just use the default COMPONENT method
34 });
35 my $meta = $controller->meta;
36 $meta->make_mutable; # Dirty, I should build the class, add the methods, then
37 # last of all make it a component
cbc455a6 38 $meta->remove_method('COMPONENT');
8483c76b 39 $meta->superclasses($app . '::ControllerBase');
77e54b00 40
41 my $methods = $model->meta->get_method_map;
42 foreach my $method_name (keys %$methods) {
43 $controller->meta->add_method(
44 # Note need to pass model name, as the method actually comes from
45 # the underlying model class, not the Catalyst shim class we autogenerated.
46 $method_name => $app->generate_reflected_controller_action_method($suffix, $methods->{$method_name})
47 );
48 }
49 $meta->make_immutable;
50}
51
52sub generate_reflected_controller_action_method {
53 my ( $app, $model, $method ) = @_;
54 my $method_name = $method->name; # Is it worth passing the actual method object here?
55 sub {
56 my ($self, $c, @args) = @_;
57 $c->res->header('X-From-Model', $model);
58 $c->res->header('X-From-Model-Data', $c->model($model)->$method_name(@args));
59 $c->res->body('OK');
60 };
59fc9d16 61}
62
631;
64