Commit | Line | Data |
b7da37bd |
1 | package Catalyst::IOC::ConstructorInjection; |
2 | use Moose; |
eff5194d |
3 | use Try::Tiny; |
9e6091e2 |
4 | use Catalyst::Utils (); |
b7da37bd |
5 | extends 'Bread::Board::ConstructorInjection'; |
6 | |
bf142143 |
7 | with 'Bread::Board::Service::WithClass', |
104155f6 |
8 | 'Bread::Board::Service::WithParameters', |
c614c3c2 |
9 | 'Bread::Board::Service::WithDependencies'; |
b7da37bd |
10 | |
9e6091e2 |
11 | has config_key => ( |
12 | is => 'ro', |
13 | isa => 'Str', |
14 | lazy_build => 1, |
15 | ); |
16 | |
c614c3c2 |
17 | sub _build_config_key { Catalyst::Utils::class2classsuffix( shift->class ) } |
18 | |
19 | sub _build_constructor_name { 'COMPONENT' } |
9e6091e2 |
20 | |
b7da37bd |
21 | sub get { |
22 | my $self = shift; |
23 | |
eff5194d |
24 | my $instance; |
25 | |
b7da37bd |
26 | my $constructor = $self->constructor_name; |
b7da37bd |
27 | my $component = $self->class; |
c614c3c2 |
28 | my $config = $self->param('config')->{ $self->config_key } || {}; |
3f1b0032 |
29 | # FIXME - Is depending on the application name to pass into constructors here a good idea? |
30 | # This makes app/ctx split harder I think.. Need to think more here, but I think |
31 | # we want to pass the application in as a parameter when building the service |
32 | # rather than depending on the app name, so that later, when the app becomes an instance |
33 | # then it'll get passed in, and components can stash themselves 'per app instance' |
c614c3c2 |
34 | my $app_name = $self->param('application_name'); |
b7da37bd |
35 | |
b7da37bd |
36 | # Stash catalyst_component_name in the config here, so that custom COMPONENT |
37 | # methods also pass it. local to avoid pointlessly shitting in config |
38 | # for the debug screen, as $component is already the key name. |
bf142143 |
39 | local $config->{catalyst_component_name} = $component; |
b7da37bd |
40 | |
c614c3c2 |
41 | unless ( $component->can( $constructor ) ) { |
42 | # FIXME - make some deprecation warnings |
43 | return $component; |
44 | } |
45 | |
eff5194d |
46 | try { |
47 | $instance = $component->$constructor( $app_name, $config ); |
c614c3c2 |
48 | } |
eff5194d |
49 | catch { |
c614c3c2 |
50 | Catalyst::Exception->throw( |
eff5194d |
51 | message => qq/Couldn't instantiate component "$component", "$_"/ |
c614c3c2 |
52 | ); |
eff5194d |
53 | }; |
54 | |
55 | return $instance |
56 | if blessed $instance; |
57 | |
58 | my $metaclass = Moose::Util::find_meta($component); |
59 | my $method_meta = $metaclass->find_method_by_name($constructor); |
60 | my $component_method_from = $method_meta->associated_metaclass->name; |
61 | my $value = defined($instance) ? $instance : 'undef'; |
62 | Catalyst::Exception->throw( |
63 | message => |
64 | qq/Couldn't instantiate component "$component", $constructor() method (from $component_method_from) didn't return an object-like value (value was $value)./ |
65 | ); |
b7da37bd |
66 | } |
67 | |
68 | __PACKAGE__->meta->make_immutable; |
69 | |
70 | no Moose; 1; |
71 | |
72 | __END__ |
73 | |
74 | =pod |
75 | |
76 | =head1 NAME |
77 | |
1ab07ed4 |
78 | Catalyst::IOC::ConstructorInjection |
79 | |
80 | =head1 SYNOPSIS |
81 | |
82 | =head1 DESCRIPTION |
b7da37bd |
83 | |
84 | =head1 AUTHORS |
85 | |
86 | Catalyst Contributors, see Catalyst.pm |
87 | |
88 | =head1 COPYRIGHT |
89 | |
90 | This library is free software. You can redistribute it and/or modify it under |
91 | the same terms as Perl itself. |
92 | |
93 | =cut |