Server : Apache/2.4.43 (Win64) OpenSSL/1.1.1g PHP/7.4.6 System : Windows NT USER-PC 6.1 build 7601 (Windows 7 Professional Edition Service Pack 1) AMD64 User : User ( 0) PHP Version : 7.4.6 Disable Function : NONE Directory : C:/xampp/perl/vendor/lib/MooseX/Method/Signatures/Meta/ |
package MooseX::Method::Signatures::Meta::Method; BEGIN { $MooseX::Method::Signatures::Meta::Method::AUTHORITY = 'cpan:FLORA'; } { $MooseX::Method::Signatures::Meta::Method::VERSION = '0.44'; } # ABSTRACT: Provides the metaclass for methods with signatures use Moose; use Carp qw/cluck/; use Context::Preserve; use Parse::Method::Signatures 1.003014; use Parse::Method::Signatures::TypeConstraint; use Scalar::Util qw/weaken/; use Moose::Util qw/does_role/; use Moose::Util::TypeConstraints; use MooseX::Meta::TypeConstraint::ForceCoercion; use MooseX::Types::Util qw/has_available_type_export/; use MooseX::Types::Structured qw/Tuple Dict Optional slurpy/; use MooseX::Types::Moose qw/ArrayRef Str Maybe Object Any CodeRef Bool/; use MooseX::Method::Signatures::Types qw/Injections Params/; use aliased 'Parse::Method::Signatures::Param::Named'; use aliased 'Parse::Method::Signatures::Param::Placeholder'; use namespace::autoclean; extends 'Moose::Meta::Method'; has signature => ( is => 'ro', isa => Str, default => '(@)', required => 1, ); has parsed_signature => ( is => 'ro', isa => class_type('Parse::Method::Signatures::Sig'), lazy => 1, builder => '_build_parsed_signature', ); sub _parsed_signature { cluck '->_parsed_signature is deprecated. use ->parsed_signature instead.'; shift->parsed_signature; } has _lexicals => ( is => 'ro', isa => ArrayRef[Str], lazy => 1, builder => '_build__lexicals', ); has injectable_code => ( is => 'ro', isa => Str, lazy => 1, builder => '_build_injectable_code', ); has _positional_args => ( is => 'ro', isa => ArrayRef, lazy => 1, builder => '_build__positional_args', ); has _named_args => ( is => 'ro', isa => ArrayRef, lazy => 1, builder => '_build__named_args', ); has _has_slurpy_positional => ( is => 'rw', isa => Bool, ); has type_constraint => ( is => 'ro', isa => class_type('Moose::Meta::TypeConstraint'), lazy => 1, builder => '_build_type_constraint', ); has return_signature => ( is => 'ro', isa => Str, predicate => 'has_return_signature', ); has _return_type_constraint => ( is => 'ro', isa => class_type('Moose::Meta::TypeConstraint'), lazy => 1, builder => '_build__return_type_constraint', ); has actual_body => ( is => 'ro', isa => CodeRef, predicate => '_has_actual_body', ); has prototype_injections => ( is => 'rw', isa => Injections, trigger => \&_parse_prototype_injections ); has _parsed_prototype_injections => ( is => 'ro', isa => Params, predicate => '_has_parsed_prototype_injections', writer => '_set_parsed_prototype_injections', ); before actual_body => sub { my ($self) = @_; confess "method doesn't have an actual body yet" unless $self->_has_actual_body; }; around name => sub { my ($next, $self) = @_; my $ret = $self->$next; confess "method doesn't have a name yet" unless defined $ret; return $ret; }; sub _wrapped_body { my ($class, $self, %args) = @_; if (exists $args{return_signature}) { return sub { my @args = ${ $self }->validate(\@_); return preserve_context { ${ $self }->actual_body->(@args) } after => sub { if (defined (my $msg = ${ $self }->_return_type_constraint->validate(\@_))) { confess $msg; } }; }; } my $actual_body; return sub { @_ = ${ $self }->validate(\@_); $actual_body ||= ${ $self }->actual_body; goto &{ $actual_body }; }; } around wrap => sub { my $orig = shift; my $self; my ($class, $code, %args); if (ref $_[1]) { ($class, $code, %args) = @_; } else { ($class, %args) = @_; $code = delete $args{body}; } my $wrapped = $class->_wrapped_body(\$self, %args); $self = $class->$orig($wrapped, %args, $code ? (actual_body => $code) : ()); # Vivify the type constraints so TC lookups happen before namespace::clean # removes them $self->type_constraint; $self->_return_type_constraint if $self->has_return_signature; weaken($self->{associated_metaclass}) if $self->{associated_metaclass}; return $self; }; sub reify { my $self = shift; my %args = @_; my %other_args = %{$self}; delete $other_args{body}; delete $other_args{actual_body}; my $body = delete $args{body} || delete $args{actual_body} || $self->body; my %final_args = (%other_args, %args); return $self->meta->name->wrap($body, %final_args); } sub _build_parsed_signature { my ($self) = @_; return Parse::Method::Signatures->signature( input => $self->signature, from_namespace => $self->package_name, ); } sub _build__return_type_constraint { my ($self) = @_; confess 'no return type constraint' unless $self->has_return_signature; my $parser = Parse::Method::Signatures->new( input => $self->return_signature, from_namespace => $self->package_name, ); my $param = $parser->_param_typed({}); confess 'failed to parse return value type constraint' unless exists $param->{type_constraints}; return Tuple[$param->{type_constraints}->tc]; } sub _param_to_spec { my ($self, $param) = @_; my $tc = Any; { # Ensure errors get reported from the right place local $Carp::Internal{'MooseX::Method::Signatures::Meta::Method'} = 1; local $Carp::Internal{'Moose::Meta::Method::Delegation'} = 1; local $Carp::Internal{'Moose::Meta::Method::Accessor'} = 1; local $Carp::Internal{'MooseX::Method::Signatures'} = 1; local $Carp::Internal{'Devel::Declare'} = 1; $tc = $param->meta_type_constraint if $param->has_type_constraints; } if ($param->has_constraints) { my $cb = join ' && ', map { "sub {${_}}->(\\\@_)" } $param->constraints; my $code = eval "sub {${cb}}"; $tc = subtype({ as => $tc, where => $code }); } my %spec; if ($param->sigil ne '$') { $spec{slurpy} = 1; $tc = slurpy ArrayRef[$tc]; } $spec{tc} = $param->required ? $tc : Optional[$tc]; $spec{default} = $param->default_value if $param->has_default_value; if ($param->has_traits) { for my $trait (@{ $param->param_traits }) { next unless $trait->[1] eq 'coerce'; $spec{coerce} = 1; } } return \%spec; } sub _parse_prototype_injections { my $self = shift; my @params; for my $inject (@{ $self->prototype_injections }) { my $param; eval { $param = Parse::Method::Signatures->param($inject); }; confess "There was a problem parsing the prototype injection '$inject': $@" if $@ || !defined $param; push @params, $param; } my @return = reverse @params; $self->_set_parsed_prototype_injections(\@return); } sub _build__lexicals { my ($self) = @_; my ($sig) = $self->parsed_signature; my @lexicals; if ($self->_has_parsed_prototype_injections) { push @lexicals, $_->variable_name for @{ $self->_parsed_prototype_injections }; } push @lexicals, $sig->has_invocant ? $sig->invocant->variable_name : '$self'; push @lexicals, (does_role($_, Placeholder) ? 'undef' : $_->variable_name) for (($sig->has_positional_params ? $sig->positional_params : ()), ($sig->has_named_params ? $sig->named_params : ())); return \@lexicals; } sub _build_injectable_code { my ($self) = @_; my $vars = join q{,}, @{ $self->_lexicals }; return "my (${vars}) = \@_;"; } sub _build__positional_args { my ($self) = @_; my $sig = $self->parsed_signature; my @positional; if ($self->_has_parsed_prototype_injections) { push @positional, map { $self->_param_to_spec($_) } @{ $self->_parsed_prototype_injections }; } push @positional, $sig->has_invocant ? $self->_param_to_spec($sig->invocant) : { tc => Object }; my $slurpy = 0; if ($sig->has_positional_params) { for my $param ($sig->positional_params) { my $spec = $self->_param_to_spec($param); $slurpy ||= 1 if $spec->{slurpy}; push @positional, $spec; } } $self->_has_slurpy_positional($slurpy); return \@positional; } sub _build__named_args { my ($self) = @_; my $sig = $self->parsed_signature; # triggering building of positionals before named params is important # because the latter needs to know if there have been any slurpy # positionals to report errors $self->_positional_args; my @named; if ($sig->has_named_params) { confess 'Named parameters cannot be combined with slurpy positionals' if $self->_has_slurpy_positional; for my $param ($sig->named_params) { push @named, $param->label => $self->_param_to_spec($param); } } return \@named; } sub _build_type_constraint { my ($self) = @_; my ($positional, $named) = map { $self->$_ } map { "_${_}_args" } qw/positional named/; my $tc = Tuple[ Tuple[ map { $_->{tc} } @{ $positional } ], Dict[ map { ref $_ ? $_->{tc} : $_ } @{ $named } ], ]; my $coerce_param = sub { my ($spec, $value) = @_; return $value unless exists $spec->{coerce}; return $spec->{tc}->coerce($value); }; my %named = @{ $named }; coerce $tc, from ArrayRef, via { my (@positional_args, %named_args); my $i = 0; for my $param (@{ $positional }) { push @positional_args, map { $coerce_param->($param, $_) } $#{ $_ } < $i ? (exists $param->{default} ? eval $param->{default} : ()) : $_->[$i]; $i++; } if (%named) { my @rest = @{ $_ }[$i .. $#{ $_ }]; confess "Expected named arguments but didn't find an even-sized list" unless @rest % 2 == 0; my %rest = @rest; while (my ($key, $spec) = each %named) { if (exists $rest{$key}) { $named_args{$key} = $coerce_param->($spec, delete $rest{$key}); next; } if (exists $spec->{default}) { $named_args{$key} = $coerce_param->($spec, eval $spec->{default}); } } @named_args{keys %rest} = values %rest; } elsif ($#{ $_ } >= $i) { push @positional_args, @{ $_ }[$i .. $#{ $_ }]; } return [\@positional_args, \%named_args]; }; return MooseX::Meta::TypeConstraint::ForceCoercion->new( type_constraint => $tc, ); } sub validate { my ($self, $args) = @_; my @named = grep { !ref $_ } @{ $self->_named_args }; my $coerced; if (defined (my $msg = $self->type_constraint->validate($args, \$coerced))) { confess $msg; } return @{ $coerced->[0] }, map { $coerced->[1]->{$_} } @named; } __PACKAGE__->meta->make_immutable; 1; __END__ =pod =encoding utf-8 =head1 NAME MooseX::Method::Signatures::Meta::Method - Provides the metaclass for methods with signatures =head1 AUTHORS =over 4 =item * Florian Ragwitz <rafl@debian.org> =item * Ash Berlin <ash@cpan.org> =item * Cory Watson <gphat@cpan.org> =item * Daniel Ruoso <daniel@ruoso.com> =item * Dave Rolsky <autarch@urth.org> =item * Hakim Cassimally <hakim.cassimally@gmail.com> =item * Jonathan Scott Duff <duff@pobox.com> =item * Justin Hunter <justin.d.hunter@gmail.com> =item * Kent Fredric <kentfredric@gmail.com> =item * Maik Hentsche <maik.hentsche@amd.com> =item * Matt Kraai <kraai@ftbfs.org> =item * Rhesa Rozendaal <rhesa@cpan.org> =item * Ricardo SIGNES <rjbs@cpan.org> =item * Steffen Schwigon <ss5@renormalist.net> =item * Yanick Champoux <yanick@babyl.dyndns.org> =item * Nicholas Perez <nperez@cpan.org> =item * Karen Etheridge <ether@cpan.org> =back =head1 COPYRIGHT AND LICENSE This software is copyright (c) 2012 by Florian Ragwitz. This is free software; you can redistribute it and/or modify it under the same terms as the Perl 5 programming language system itself. =cut