Add explicit return values for (almost) all native delegation mutating methods
[gitmo/Moose.git] / t / 070_native_traits / 060_trait_number.t
1 #!/usr/bin/perl
2
3 use strict;
4 use warnings;
5
6 use lib 't/lib';
7
8 use Moose ();
9 use Moose::Util::TypeConstraints;
10 use NoInlineAttribute;
11 use Test::Exception;
12 use Test::More;
13 use Test::Moose;
14
15 {
16     my %handles = (
17         abs         => 'abs',
18         add         => 'add',
19         inc         => [ add => 1 ],
20         div         => 'div',
21         cut_in_half => [ div => 2 ],
22         mod         => 'mod',
23         odd         => [ mod => 2 ],
24         mul         => 'mul',
25         set         => 'set',
26         sub         => 'sub',
27         dec         => [ sub => 1 ],
28     );
29
30     my $name = 'Foo1';
31
32     sub build_class {
33         my %attr = @_;
34
35         my $class = Moose::Meta::Class->create(
36             $name++,
37             superclasses => ['Moose::Object'],
38         );
39
40         my @traits = 'Number';
41         push @traits, 'NoInlineAttribute'
42             if delete $attr{no_inline};
43
44         $class->add_attribute(
45             integer => (
46                 traits  => \@traits,
47                 is      => 'ro',
48                 isa     => 'Int',
49                 default => 5,
50                 handles => \%handles,
51                 clearer => '_clear_integer',
52                 %attr,
53             ),
54         );
55
56         return ( $class->name, \%handles );
57     }
58 }
59
60 {
61     run_tests(build_class);
62     run_tests( build_class( lazy => 1 ) );
63     run_tests( build_class( trigger => sub { } ) );
64     run_tests( build_class( no_inline => 1 ) );
65
66     # Will force the inlining code to check the entire hashref when it is modified.
67     subtype 'MyInt', as 'Int', where { 1 };
68
69     run_tests( build_class( isa => 'MyInt' ) );
70
71     coerce 'MyInt', from 'Int', via { $_ };
72
73     run_tests( build_class( isa => 'MyInt', coerce => 1 ) );
74 }
75
76 sub run_tests {
77     my ( $class, $handles ) = @_;
78
79     can_ok( $class, $_ ) for sort keys %{$handles};
80
81     with_immutable {
82         my $obj = $class->new;
83
84         is( $obj->integer, 5, 'Default to five' );
85
86         is( $obj->add(10), 15, 'add returns new value' );
87
88         is( $obj->integer, 15, 'Add ten for fithteen' );
89
90         throws_ok { $obj->add( 10, 2 ) }
91         qr/Cannot call add with more than 1 argument/,
92             'add throws an error when 2 arguments are passed';
93
94         is( $obj->sub(3), 12, 'sub returns new value' );
95
96         is( $obj->integer, 12, 'Subtract three for 12' );
97
98         throws_ok { $obj->sub( 10, 2 ) }
99         qr/Cannot call sub with more than 1 argument/,
100             'sub throws an error when 2 arguments are passed';
101
102         is( $obj->set(10), 10, 'set returns new value' );
103
104         is( $obj->integer, 10, 'Set to ten' );
105
106         throws_ok { $obj->set( 10, 2 ) }
107         qr/Cannot call set with more than 1 argument/,
108             'set throws an error when 2 arguments are passed';
109
110         is( $obj->div(2), 5, 'div returns new value' );
111
112         is( $obj->integer, 5, 'divide by 2' );
113
114         throws_ok { $obj->div( 10, 2 ) }
115         qr/Cannot call div with more than 1 argument/,
116             'div throws an error when 2 arguments are passed';
117
118         is( $obj->mul(2), 10, 'mul returns new value' );
119
120         is( $obj->integer, 10, 'multiplied by 2' );
121
122         throws_ok { $obj->mul( 10, 2 ) }
123         qr/Cannot call mul with more than 1 argument/,
124             'mul throws an error when 2 arguments are passed';
125
126         is( $obj->mod(2), 0, 'mod returns new value' );
127
128         is( $obj->integer, 0, 'Mod by 2' );
129
130         throws_ok { $obj->mod( 10, 2 ) }
131         qr/Cannot call mod with more than 1 argument/,
132             'mod throws an error when 2 arguments are passed';
133
134         $obj->set(7);
135
136         $obj->mod(5);
137
138         is( $obj->integer, 2, 'Mod by 5' );
139
140         $obj->set(-1);
141
142         is( $obj->abs, 1, 'abs returns new value' );
143
144         throws_ok { $obj->abs(10) }
145         qr/Cannot call abs with any arguments/,
146             'abs throws an error when an argument is passed';
147
148         is( $obj->integer, 1, 'abs 1' );
149
150         $obj->set(12);
151
152         $obj->inc;
153
154         is( $obj->integer, 13, 'inc 12' );
155
156         $obj->dec;
157
158         is( $obj->integer, 12, 'dec 13' );
159
160         if ( $class->meta->get_attribute('integer')->is_lazy ) {
161             my $obj = $class->new;
162
163             $obj->add(2);
164
165             is( $obj->integer, 7, 'add with lazy default' );
166
167             $obj->_clear_integer;
168
169             $obj->mod(2);
170
171             is( $obj->integer, 1, 'mod with lazy default' );
172         }
173     }
174     $class;
175 }
176
177 done_testing;