X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?a=blobdiff_plain;f=lib%2FMoose%2FMeta%2FTypeConstraint.pm;h=05a3f440fd247d88ed6a371a4a03d88435f45496;hb=b3853db9543d3f4da31a681df2300fc2de3c3077;hp=4592a638a79494286a53fb1b938bbf50f290b7cd;hpb=547dda77201ee580b3dfda83b95e278ce6232182;p=gitmo%2FMoose.git diff --git a/lib/Moose/Meta/TypeConstraint.pm b/lib/Moose/Meta/TypeConstraint.pm index 4592a63..05a3f44 100644 --- a/lib/Moose/Meta/TypeConstraint.pm +++ b/lib/Moose/Meta/TypeConstraint.pm @@ -8,11 +8,12 @@ use metaclass; use overload '""' => sub { shift->name }, # stringify to tc name fallback => 1; -use Sub::Name 'subname'; -use Carp 'confess'; -use Scalar::Util 'blessed'; +use Scalar::Util qw(blessed refaddr); -our $VERSION = '0.12'; +use base qw(Class::MOP::Object); + +our $VERSION = '0.59'; +$VERSION = eval $VERSION; our $AUTHORITY = 'cpan:STEVAN'; __PACKAGE__->meta->add_attribute('name' => (reader => 'name')); @@ -20,10 +21,12 @@ __PACKAGE__->meta->add_attribute('parent' => ( reader => 'parent', predicate => 'has_parent', )); + +my $null_constraint = sub { 1 }; __PACKAGE__->meta->add_attribute('constraint' => ( reader => 'constraint', writer => '_set_constraint', - default => sub { sub { 1 } } + default => sub { $null_constraint } )); __PACKAGE__->meta->add_attribute('message' => ( accessor => 'message', @@ -39,6 +42,11 @@ __PACKAGE__->meta->add_attribute('hand_optimized_type_constraint' => ( predicate => 'has_hand_optimized_type_constraint', )); +sub parents { + my $self; + $self->parent; +} + # private accessors __PACKAGE__->meta->add_attribute('compiled_type_constraint' => ( @@ -51,13 +59,13 @@ __PACKAGE__->meta->add_attribute('package_defined_in' => ( sub new { my $class = shift; - my $self = $class->meta->new_object(@_); + my $self = $class->_new(@_); $self->compile_type_constraint() unless $self->_has_compiled_type_constraint; return $self; } -sub coerce { ((shift)->coercion || confess "Cannot coerce without a type coercion")->coerce(@_) } +sub coerce { ((shift)->coercion || Moose->throw_error("Cannot coerce without a type coercion"))->coerce(@_) } sub check { $_[0]->_compiled_type_constraint->($_[1]) ? 1 : undef } sub validate { my ($self, $value) = @_; @@ -71,30 +79,61 @@ sub validate { sub get_message { my ($self, $value) = @_; - $value = (defined $value ? overload::StrVal($value) : 'undef'); if (my $msg = $self->message) { local $_ = $value; return $msg->($value); } else { + $value = (defined $value ? overload::StrVal($value) : 'undef'); return "Validation failed for '" . $self->name . "' failed with value $value"; } } ## type predicates ... +sub equals { + my ( $self, $type_or_name ) = @_; + + my $other = Moose::Util::TypeConstraints::find_type_constraint($type_or_name) or return; + + return 1 if refaddr($self) == refaddr($other); + + if ( $self->has_hand_optimized_type_constraint and $other->has_hand_optimized_type_constraint ) { + return 1 if $self->hand_optimized_type_constraint == $other->hand_optimized_type_constraint; + } + + return unless $self->constraint == $other->constraint; + + if ( $self->has_parent ) { + return unless $other->has_parent; + return unless $self->parent->equals( $other->parent ); + } else { + return if $other->has_parent; + } + + return 1; +} + sub is_a_type_of { - my ($self, $type_name) = @_; - ($self->name eq $type_name || $self->is_subtype_of($type_name)); + my ($self, $type_or_name) = @_; + + my $type = Moose::Util::TypeConstraints::find_type_constraint($type_or_name) or return; + + ($self->equals($type) || $self->is_subtype_of($type)); } sub is_subtype_of { - my ($self, $type_name) = @_; + my ($self, $type_or_name) = @_; + + my $type = Moose::Util::TypeConstraints::find_type_constraint($type_or_name) or return; + my $current = $self; + while (my $parent = $current->parent) { - return 1 if $parent->name eq $type_name; + return 1 if $parent->equals($type); $current = $parent; } + return 0; } @@ -115,9 +154,9 @@ sub _actually_compile_type_constraint { my $check = $self->constraint; (defined $check) - || confess "Could not compile type constraint '" + || Moose->throw_error("Could not compile type constraint '" . $self->name - . "' because no constraint check"; + . "' because no constraint check"); return $self->_compile_subtype($check) if $self->has_parent; @@ -130,7 +169,7 @@ sub _compile_hand_optimized_type_constraint { my $type_constraint = $self->hand_optimized_type_constraint; - confess unless ref $type_constraint; + Moose->throw_error("Hand optimized type constraint is not a code reference") unless ref $type_constraint; return $type_constraint; } @@ -138,38 +177,60 @@ sub _compile_hand_optimized_type_constraint { sub _compile_subtype { my ($self, $check) = @_; - # so we gather all the parents in order - # and grab their constraints ... + # gather all the parent constraintss in order my @parents; + my $optimized_parent; foreach my $parent ($self->_collect_all_parents) { + # if a parent is optimized, the optimized constraint already includes + # all of its parents tcs, so we can break the loop if ($parent->has_hand_optimized_type_constraint) { - unshift @parents => $parent->hand_optimized_type_constraint; + push @parents => $optimized_parent = $parent->hand_optimized_type_constraint; last; } else { - unshift @parents => $parent->constraint; + push @parents => $parent->constraint; } } - # then we compile them to run without - # having to recurse as we did before - return subname $self->name => sub { - local $_ = $_[0]; - foreach my $parent (@parents) { - return undef unless $parent->($_[0]); + @parents = grep { $_ != $null_constraint } reverse @parents; + + unless ( @parents ) { + return $self->_compile_type($check); + } elsif( $optimized_parent and @parents == 1 ) { + # the case of just one optimized parent is optimized to prevent + # looping and the unnecessary localization + if ( $check == $null_constraint ) { + return $optimized_parent; + } else { + return Class::MOP::subname($self->name, sub { + return undef unless $optimized_parent->($_[0]); + local $_ = $_[0]; + $check->($_[0]); + }); } - return undef unless $check->($_[0]); - 1; - }; + } else { + # general case, check all the constraints, from the first parent to ourselves + my @checks = @parents; + push @checks, $check if $check != $null_constraint; + return Class::MOP::subname($self->name => sub { + local $_ = $_[0]; + foreach my $check (@checks) { + return undef unless $check->($_[0]); + } + return 1; + }); + } } sub _compile_type { my ($self, $check) = @_; - return subname $self->name => sub { + + return $check if $check == $null_constraint; # Item, Any + + return Class::MOP::subname($self->name => sub { local $_ = $_[0]; - return undef unless $check->($_[0]); - 1; - }; + $check->($_[0]); + }); } ## other utils ... @@ -185,6 +246,12 @@ sub _collect_all_parents { return @parents; } +sub create_childtype { + my ($self, %opts) = @_; + my $class = ref $self; + return $class->new(%opts, parent => $self); +} + ## this should get deprecated actually ... sub union { Carp::croak "DEPRECATED" } @@ -218,12 +285,25 @@ If you wish to use features at this depth, please come to the =item B -=item B +=item B -This checks the current type name, and if it does not match, -checks if it is a subtype of it. +This checks the current type against the supplied type (only). +Returns false if the two types are not equal. It also returns false if +you provide the type as a name, and the type name isn't found in the +type registry. -=item B +=item B + +This checks the current type against the supplied type, or if the +current type is a sub-type of the type name or object supplied. It +also returns false if you provide the type as a name, and the type +name isn't found in the type registry. + +=item B + +This checks the current type is a sub-type of the type name or object +supplied. It also returns false if you provide the type as a name, and +the type name isn't found in the type registry. =item B @@ -245,10 +325,18 @@ the C will be used to construct a custom error message. =item B +The name of the type in the global type registry. + =item B +This type's parent type. + =item B +Returns true if this type has a parent type. + +=item B + =item B =item B @@ -265,6 +353,8 @@ the C will be used to construct a custom error message. =item B +=item B + =back =head2 DEPRECATED METHOD