Remove numbers from our tests
[gitmo/Moose.git] / t / native_traits / trait_hash.t
diff --git a/t/native_traits/trait_hash.t b/t/native_traits/trait_hash.t
new file mode 100644 (file)
index 0000000..107b851
--- /dev/null
@@ -0,0 +1,295 @@
+#!/usr/bin/perl
+
+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 = (
+        option_accessor  => 'accessor',
+        quantity         => [ accessor => 'quantity' ],
+        clear_options    => 'clear',
+        num_options      => 'count',
+        delete_option    => 'delete',
+        is_defined       => 'defined',
+        options_elements => 'elements',
+        has_option       => 'exists',
+        get_option       => 'get',
+        has_no_options   => 'is_empty',
+        keys             => 'keys',
+        values           => 'values',
+        key_value        => 'kv',
+        set_option       => 'set',
+    );
+
+    my $name = 'Foo1';
+
+    sub build_class {
+        my %attr = @_;
+
+        my $class = Moose::Meta::Class->create(
+            $name++,
+            superclasses => ['Moose::Object'],
+        );
+
+        my @traits = 'Hash';
+        push @traits, 'NoInlineAttribute'
+            if delete $attr{no_inline};
+
+        $class->add_attribute(
+            options => (
+                traits  => \@traits,
+                is      => 'rw',
+                isa     => 'HashRef[Str]',
+                default => sub { {} },
+                handles => \%handles,
+                clearer => '_clear_options',
+                %attr,
+            ),
+        );
+
+        return ( $class->name, \%handles );
+    }
+}
+
+{
+    run_tests(build_class);
+    run_tests( build_class( lazy => 1, default => sub { { x => 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 'MyHashRef', as 'HashRef[Str]', where { 1 };
+
+    run_tests( build_class( isa => 'MyHashRef' ) );
+
+    coerce 'MyHashRef', from 'HashRef', via { $_ };
+
+    run_tests( build_class( isa => 'MyHashRef', coerce => 1 ) );
+}
+
+sub run_tests {
+    my ( $class, $handles ) = @_;
+
+    can_ok( $class, $_ ) for sort keys %{$handles};
+
+    with_immutable {
+        my $obj = $class->new( options => {} );
+
+        ok( $obj->has_no_options, '... we have no options' );
+        is( $obj->num_options, 0, '... we have no options' );
+
+        is_deeply( $obj->options, {}, '... no options yet' );
+        ok( !$obj->has_option('foo'), '... we have no foo option' );
+
+        is( exception {
+            is(
+                $obj->set_option( foo => 'bar' ),
+                'bar',
+                'set return single new value in scalar context'
+            );
+        }, undef, '... set the option okay' );
+
+        ok( $obj->is_defined('foo'), '... foo is defined' );
+
+        ok( !$obj->has_no_options, '... we have options' );
+        is( $obj->num_options, 1, '... we have 1 option(s)' );
+        ok( $obj->has_option('foo'), '... we have a foo option' );
+        is_deeply( $obj->options, { foo => 'bar' }, '... got options now' );
+
+        is( exception {
+            $obj->set_option( bar => 'baz' );
+        }, undef, '... set the option okay' );
+
+        is( $obj->num_options, 2, '... we have 2 option(s)' );
+        is_deeply(
+            $obj->options, { foo => 'bar', bar => 'baz' },
+            '... got more options now'
+        );
+
+        is( $obj->get_option('foo'), 'bar', '... got the right option' );
+
+        is_deeply(
+            [ $obj->get_option(qw(foo bar)) ], [qw(bar baz)],
+            "get multiple options at once"
+        );
+
+        is(
+            scalar( $obj->get_option(qw( foo bar)) ), "baz",
+            '... got last option in scalar context'
+        );
+
+        is( exception {
+            $obj->set_option( oink => "blah", xxy => "flop" );
+        }, undef, '... set the option okay' );
+
+        is( $obj->num_options, 4, "4 options" );
+        is_deeply(
+            [ $obj->get_option(qw(foo bar oink xxy)) ],
+            [qw(bar baz blah flop)], "get multiple options at once"
+        );
+
+        is( exception {
+            is( scalar $obj->delete_option('bar'), 'baz',
+                'delete returns deleted value' );
+        }, undef, '... deleted the option okay' );
+
+        is( exception {
+            is_deeply(
+                [ $obj->delete_option( 'oink', 'xxy' ) ],
+                [ 'blah', 'flop' ],
+                'delete returns all deleted values in list context'
+            );
+        }, undef, '... deleted multiple option okay' );
+
+        is( $obj->num_options, 1, '... we have 1 option(s)' );
+        is_deeply(
+            $obj->options, { foo => 'bar' },
+            '... got more options now'
+        );
+
+        $obj->clear_options;
+
+        is_deeply( $obj->options, {}, "... cleared options" );
+
+        is( exception {
+            $obj->quantity(4);
+        }, undef, '... options added okay with defaults' );
+
+        is( $obj->quantity, 4, 'reader part of curried accessor works' );
+
+        is(
+            $obj->option_accessor('quantity'), 4,
+            'accessor as reader'
+        );
+
+        is_deeply(
+            $obj->options, { quantity => 4 },
+            '... returns what we expect'
+        );
+
+        $obj->option_accessor( size => 42 );
+
+        like(
+            exception {
+                $obj->option_accessor;
+            },
+            qr/Cannot call accessor without at least 1 argument/,
+            'error when calling accessor with no arguments'
+        );
+
+        is_deeply(
+            $obj->options, { quantity => 4, size => 42 },
+            'accessor as writer'
+        );
+
+        is( exception {
+            $class->new( options => { foo => 'BAR' } );
+        }, undef, '... good constructor params' );
+
+        isnt( exception {
+            $obj->set_option( bar => {} );
+        }, undef, '... could not add a hash ref where an string is expected' );
+
+        isnt( exception {
+            $class->new( options => { foo => [] } );
+        }, undef, '... bad constructor params' );
+
+        $obj->options( {} );
+
+        is_deeply(
+            [ $obj->set_option( oink => "blah", xxy => "flop" ) ],
+            [ 'blah', 'flop' ],
+            'set returns newly set values in order of keys provided'
+        );
+
+        is_deeply(
+            [ sort $obj->keys ],
+            [ 'oink', 'xxy' ],
+            'keys returns expected keys'
+        );
+
+        is_deeply(
+            [ sort $obj->values ],
+            [ 'blah', 'flop' ],
+            'values returns expected values'
+        );
+
+        my @key_value = sort { $a->[0] cmp $b->[0] } $obj->key_value;
+        is_deeply(
+            \@key_value,
+            [
+                sort { $a->[0] cmp $b->[0] }[ 'xxy', 'flop' ],
+                [ 'oink',     'blah' ]
+            ],
+            '... got the right key value pairs'
+            )
+            or do {
+            require Data::Dumper;
+            diag( Data::Dumper::Dumper( \@key_value ) );
+            };
+
+        my %options_elements = $obj->options_elements;
+        is_deeply(
+            \%options_elements, {
+                'oink'     => 'blah',
+                'xxy'      => 'flop'
+            },
+            '... got the right hash elements'
+        );
+
+        if ( $class->meta->get_attribute('options')->is_lazy ) {
+            my $obj = $class->new;
+
+            $obj->set_option( y => 2 );
+
+            is_deeply(
+                $obj->options, { x => 1, y => 2 },
+                'set_option with lazy default'
+            );
+
+            $obj->_clear_options;
+
+            ok(
+                $obj->has_option('x'),
+                'key for x exists - lazy default'
+            );
+
+            $obj->_clear_options;
+
+            ok(
+                $obj->is_defined('x'),
+                'key for x is defined - lazy default'
+            );
+
+            $obj->_clear_options;
+
+            is_deeply(
+                [ $obj->key_value ],
+                [ [ x => 1 ] ],
+                'kv returns lazy default'
+            );
+
+            $obj->_clear_options;
+
+            $obj->option_accessor( y => 2 );
+
+            is_deeply(
+                [ sort $obj->keys ],
+                [ 'x', 'y' ],
+                'accessor triggers lazy default generator'
+            );
+        }
+    }
+    $class;
+}
+
+done_testing;