c443ec574ac5d9c1b2e8afe8e551a88c37766a9a
[gitmo/Moose.git] / t / basics / rebless.t
1 #!/usr/bin/perl
2
3 use strict;
4 use warnings;
5
6 use Test::More;
7 use Test::Fatal;
8 use Test::Moose qw(with_immutable);
9 use Scalar::Util 'blessed';
10
11 use Moose::Util::TypeConstraints;
12
13 subtype 'Positive'
14      => as 'Num'
15      => where { $_ > 0 };
16
17 {
18     package Parent;
19     use Moose;
20
21     has name => (
22         is       => 'rw',
23         isa      => 'Str',
24     );
25
26     has lazy_classname => (
27         is      => 'ro',
28         lazy    => 1,
29         default => sub { "Parent" },
30     );
31
32     has type_constrained => (
33         is      => 'rw',
34         isa     => 'Num',
35         default => 5.5,
36     );
37
38     package Child;
39     use Moose;
40     extends 'Parent';
41
42     has '+name' => (
43         default => 'Junior',
44     );
45
46     has '+lazy_classname' => (
47         default => sub { "Child" },
48     );
49
50     has '+type_constrained' => (
51         isa     => 'Int',
52         default => 100,
53     );
54
55     our %trigger_calls;
56     our %initializer_calls;
57
58     has new_attr => (
59         is => 'rw', isa => 'Str',
60         trigger => sub {
61             my ($self, $val, $attr) = @_;
62             $trigger_calls{new_attr}++;
63         },
64         initializer => sub {
65             my ($self, $value, $set, $attr) = @_;
66             $initializer_calls{new_attr}++;
67             $set->($value);
68         },
69     );
70 }
71
72 my @classes = qw(Parent Child);
73
74 with_immutable
75 {
76     my $foo = Parent->new;
77     my $bar = Parent->new;
78
79     is(blessed($foo), 'Parent', 'Parent->new gives a Parent object');
80     is($foo->name, undef, 'No name yet');
81     is($foo->lazy_classname, 'Parent', "lazy attribute initialized");
82     is( exception { $foo->type_constrained(10.5) }, undef, "Num type constraint for now.." );
83
84     # try to rebless, except it will fail due to Child's stricter type constraint
85     like( exception { Child->meta->rebless_instance($foo) }, qr/^Attribute \(type_constrained\) does not pass the type constraint because\: Validation failed for 'Int' with value 10\.5/, '... this failed because of type check' );
86     like( exception { Child->meta->rebless_instance($bar) }, qr/^Attribute \(type_constrained\) does not pass the type constraint because\: Validation failed for 'Int' with value 5\.5/, '... this failed because of type check' );
87
88     $foo->type_constrained(10);
89     $bar->type_constrained(5);
90
91     Child->meta->rebless_instance($foo);
92     Child->meta->rebless_instance($bar, new_attr => 'blah');
93
94     is(blessed($foo), 'Child', 'successfully reblessed into Child');
95     is($foo->name, 'Junior', "Child->name's default came through");
96
97     is($foo->lazy_classname, 'Parent', "lazy attribute was already initialized");
98     is($bar->lazy_classname, 'Child', "lazy attribute just now initialized");
99
100     like( exception { $foo->type_constrained(10.5) }, qr/^Attribute \(type_constrained\) does not pass the type constraint because\: Validation failed for 'Int' with value 10\.5/, '... this failed because of type check' );
101
102
103     is_deeply(\%Child::trigger_calls, { new_attr => 1 }, 'Trigger fired on rebless_instance');
104     is_deeply(\%Child::initializer_calls, { new_attr => 1 }, 'Initializer fired on rebless_instance');
105
106     undef %Child::trigger_calls;
107     undef %Child::initializer_calls;
108
109 }
110 @classes;
111
112 done_testing;