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