Use t::Exception for this test
[gitmo/Mouse.git] / lib / Mouse / Meta / Method / Constructor.pm
CommitLineData
fc1d8369 1package Mouse::Meta::Method::Constructor;
2use strict;
3use warnings;
4
5sub generate_constructor_method_inline {
41cdacce 6 my ($class, $meta) = @_;
24ad3f66 7
fc1d8369 8 my $buildall = $class->_generate_BUILDALL($meta);
9 my $buildargs = $class->_generate_BUILDARGS();
fc1d8369 10 my $processattrs = $class->_generate_processattrs($meta);
11
24ad3f66 12 my $code = <<"...";
fc1d8369 13 sub {
14 my \$class = shift;
15 my \$args = $buildargs;
24ad3f66 16 my \$instance = bless {}, \$class;
fc1d8369 17 $processattrs;
18 $buildall;
19 return \$instance;
20 }
21...
24ad3f66 22
23 warn $code if $ENV{DEBUG};
24
25 local $@;
26 my $res = eval $code;
27 die $@ if $@;
28 $res;
fc1d8369 29}
30
31sub _generate_processattrs {
32 my ($class, $meta, ) = @_;
33 my @attrs = $meta->compute_all_applicable_attributes;
34 my @res;
35 for my $attr (@attrs) {
36 my $from = $attr->init_arg;
37 my $key = $attr->name;
38 my $part1 = do {
39 my @code;
40 if ($attr->should_coerce) {
41 push @code, "\$args->{\$from} = \$attr->coerce_constraint( \$args->{\$from} );";
42 }
43 if ($attr->has_type_constraint) {
44 push @code, "\$attr->verify_type_constraint( \$args->{\$from} );";
45 }
46 push @code, "\$instance->{\$key} = \$args->{\$from};";
41cdacce 47 if ($attr->is_weak_ref) {
48 push @code, "weaken( \$instance->{\$key} ) if ref( \$instance->{\$key} );";
49 }
fc1d8369 50 if ( $attr->has_trigger ) {
51 push @code, "\$attr->trigger->( \$instance, \$args->{\$from}, \$attr );";
52 }
53 join "\n", @code;
54 };
55 my $part2 = do {
56 my @code;
57 if ( $attr->has_default || $attr->has_builder ) {
58 unless ( $attr->is_lazy ) {
59 my $default = $attr->default;
60 my $builder = $attr->builder;
61 if ($attr->has_builder) {
62 push @code, "my \$value = \$instance->$builder;";
63 } elsif (ref($default) eq 'CODE') {
64 push @code, "my \$value = \$attr->default()->();";
65 } else {
66 push @code, "my \$value = \$attr->default();";
67 }
68 if ($attr->should_coerce) {
69 push @code, "\$value = \$attr->coerce_constraint(\$value);";
70 }
71 if ($attr->has_type_constraint) {
72 push @code, "\$attr->verify_type_constraint(\$value);";
73 }
74 push @code, "\$instance->{\$key} = \$value;";
75 if ($attr->is_weak_ref) {
76 push @code, "weaken( \$instance->{\$key} ) if ref( \$instance->{\$key} );";
77 }
78 }
79 join "\n", @code;
80 }
81 else {
82 if ( $attr->is_required ) {
83 q{Carp::confess("Attribute (} . $attr->name . q{) is required");};
84 } else {
85 ""
86 }
87 }
88 };
89 my $code = <<"...";
90 {
41cdacce 91 my \$attr = \$meta->get_attribute('$key');
fc1d8369 92 my \$from = '$from';
93 my \$key = '$key';
94 if (defined(\$from) && exists(\$args->{\$from})) {
95 $part1;
96 } else {
97 $part2;
98 }
99 }
100...
101 push @res, $code;
102 }
103 return join "\n", @res;
104}
105
106sub _generate_BUILDARGS {
107 <<'...';
108 do {
109 if ( scalar @_ == 1 ) {
110 if ( defined $_[0] ) {
111 ( ref( $_[0] ) eq 'HASH' )
112 || Carp::confess "Single parameters to new() must be a HASH ref";
113 +{ %{ $_[0] } };
114 }
115 else {
116 +{};
117 }
118 }
119 else {
120 +{@_};
121 }
122 };
123...
124}
125
126sub _generate_BUILDALL {
127 my ($class, $meta) = @_;
128 return '' unless $meta->name->can('BUILD');
129
130 my @code = ();
131 push @code, q{no strict 'refs';};
132 push @code, q{no warnings 'once';};
133 no strict 'refs';
134 for my $class ($meta->linearized_isa) {
135 if (*{ $class . '::BUILD' }{CODE}) {
136 push @code, qq{${class}::BUILD->(\$instance, \$args);};
137 }
138 }
139 return join "\n", @code;
140}
141
1421;