From: Daisuke Maki Date: Sat, 7 Mar 2009 09:07:25 +0000 (+0000) Subject: Make coercion work for parameterized types X-Git-Tag: 0.19~5 X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?a=commitdiff_plain;h=310ad28b79e8761d5f8b24e54e06d3ebd3ad59b7;p=gitmo%2FMouse.git Make coercion work for parameterized types --- diff --git a/lib/Mouse/Util/TypeConstraints.pm b/lib/Mouse/Util/TypeConstraints.pm index e811ad5..6410243 100644 --- a/lib/Mouse/Util/TypeConstraints.pm +++ b/lib/Mouse/Util/TypeConstraints.pm @@ -90,7 +90,13 @@ sub subtype { if ($TYPE{$name} && $TYPE_SOURCE{$name} ne $pkg) { Carp::croak "The type constraint '$name' has already been created in $TYPE_SOURCE{$name} and cannot be created again in $pkg"; }; - my $constraint = $conf{where} || do { $TYPE{delete $conf{as} || 'Any' } }; + my $constraint = $conf{where} || do { + my $as = delete $conf{as} || 'Any'; + if (! exists $TYPE{$as}) { # Perhaps it's a parameterized source? + Mouse::Meta::Attribute::_build_type_constraint($as); + } + $TYPE{$as}; + }; my $as = $conf{as} || ''; $TYPE_SOURCE{$name} = $pkg; @@ -116,8 +122,14 @@ sub coerce { Carp::croak "A coercion action already exists for '$type'" if $COERCE{$name}->{$type}; - Carp::croak "Could not find the type constraint ($type) to coerce from" - unless $TYPE{$type}; + if (! $TYPE{$type}) { + # looks parameterized + if ($type =~ /^[^\[]+\[.+\]$/) { + Mouse::Meta::Attribute::_build_type_constraint($type); + } else { + Carp::croak "Could not find the type constraint ($type) to coerce from" + } + } push @{ $COERCE_KEYS{$name} }, $type; $COERCE{$name}->{$type} = $code; diff --git a/t/043-parameterized-type.t b/t/043-parameterized-type.t index 7bc3edb..8c20411 100644 --- a/t/043-parameterized-type.t +++ b/t/043-parameterized-type.t @@ -1,7 +1,7 @@ #!/usr/bin/env perl use strict; use warnings; -use Test::More tests => 7; +use Test::More tests => 9; use Test::Exception; { @@ -52,5 +52,39 @@ use Test::Exception; } qr/Attribute \(complex\) does not pass the type constraint because: Validation failed for 'ArrayRef\[HashRef\[Int\]\]' failed with value/, "Bad args for complex types throws an exception"; } +{ + { + package Bar; + use Mouse; + use Mouse::Util::TypeConstraints; + + subtype 'Bar::List' + => as 'ArrayRef[HashRef]' + ; + coerce 'Bar::List' + => from 'ArrayRef[Str]' + => via { + [ map { +{ $_ => 1 } } @$_ ] + } + ; + has 'list' => ( + is => 'ro', + isa => 'Bar::List', + coerce => 1, + ); + } + + lives_and { + my @list = ( {a => 1}, {b => 1}, {c => 1} ); + my $bar = Bar->new(list => [ qw(a b c) ]); + + is_deeply( $bar->list, \@list, "list is as expected"); + } "coercion works"; + + throws_ok { + Bar->new(list => [ { 1 => 2 }, 2, 3 ]); + } qr/Attribute \(list\) does not pass the type constraint because: Validation failed for 'Bar::List' failed with value/, "Bad coercion parameter throws an error"; +} +