Fix the logic somewhat
[catagits/Catalyst-Runtime.git] / lib / Catalyst / IOC / ConstructorInjection.pm
CommitLineData
b7da37bd 1package Catalyst::IOC::ConstructorInjection;
2use Moose;
eff5194d 3use Try::Tiny;
9e6091e2 4use Catalyst::Utils ();
b7da37bd 5extends 'Bread::Board::ConstructorInjection';
6
bf142143 7with 'Bread::Board::Service::WithClass',
104155f6 8 'Bread::Board::Service::WithParameters',
c614c3c2 9 'Bread::Board::Service::WithDependencies';
b7da37bd 10
9e6091e2 11has config_key => (
12 is => 'ro',
13 isa => 'Str',
14 lazy_build => 1,
15);
16
c614c3c2 17sub _build_config_key { Catalyst::Utils::class2classsuffix( shift->class ) }
18
19sub _build_constructor_name { 'COMPONENT' }
9e6091e2 20
b7da37bd 21sub 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
70no Moose; 1;
71
72__END__
73
74=pod
75
76=head1 NAME
77
1ab07ed4 78Catalyst::IOC::ConstructorInjection
79
80=head1 SYNOPSIS
81
82=head1 DESCRIPTION
b7da37bd 83
84=head1 AUTHORS
85
86Catalyst Contributors, see Catalyst.pm
87
88=head1 COPYRIGHT
89
90This library is free software. You can redistribute it and/or modify it under
91the same terms as Perl itself.
92
93=cut