Acme-Has-Tiny
view release on metacpan or search on metacpan
lib/Acme/Has/Tiny.pm view on Meta::CPAN
package Acme::Has::Tiny;
use 5.008;
use strict;
use warnings;
no warnings qw(uninitialized once void numeric);
our $AUTHORITY = "cpan:TOBYINK";
our $VERSION = "0.002";
use B qw(perlstring);
use Scalar::Util qw(blessed);
our %ATTRIBUTES;
our %VALIDATORS;
sub _croak ($;@)
{
my $msg = shift;
require Carp;
$Carp::CarpInternal{+__PACKAGE__} = 1;
Carp::croak(sprintf($msg, @_));
}
BEGIN { *CAN_HAZ_XS = eval 'use Class::XSAccessor 1.18; 1' ? sub(){!!1} : sub(){!!0} };
sub import
{
no strict qw(refs);
my $me = shift;
my $caller = caller;
my %want = map +($_ => 1), @_;
if ($want{has})
{
*{"$caller\::has"} = sub { unshift @_, __PACKAGE__; goto \&has };
}
if ($want{new})
{
*{"$caller\::new"} = sub {
my $new = $me->create_constructor("new", class => $_[0], replace => 1);
goto $new;
};
}
return;
}
sub has
{
my $me = shift;
my ($attrs, %options) = @_;
$attrs = [$attrs] unless ref($attrs) eq q(ARRAY);
my $class = $options{class} || caller;
delete $VALIDATORS{$class};
my @code = "package $class;";
for my $a (@$attrs)
{
$ATTRIBUTES{$class}{$a} = +{ %options };
push @code, $me->_build_methods($class, $a, $ATTRIBUTES{$class}{$a});
}
my $str = join "\n", @code, "1;";
eval($str) or die("COMPILE ERROR: $@\nCODE:\n$str\n");
return;
}
sub assert_valid
{
my $me = shift;
my ($class, $hash) = @_;
my @validator = map {
$VALIDATORS{$_} ||= $me->_compile_validator($_, $ATTRIBUTES{$_});
} $me->_find_parents($class);
$_->($hash) for @validator;
return $hash;
}
my $default_buildargs = sub
{
my $class = shift;
return +{
(@_ == 1 && ref($_[0]) eq q(HASH)) ? %{$_[0]} : @_
};
};
sub create_constructor
{
my $me = shift;
my ($method, %options) = @_;
my $class = $options{class} || caller;
my $build = $options{build};
my $buildargs = $options{buildargs} || $default_buildargs;
my $code = sub
{
my $class = shift;
my $self = bless($class->$buildargs(@_), $class);
$me->assert_valid($class, $self);
$self->$build if $options{build};
return $self;
};
no strict qw(refs);
if ($options{replace})
{
no warnings qw(redefine);
*{"$class\::$method"} = $code;
}
else
{
use warnings FATAL => qw(redefine);
*{"$class\::$method"} = $code;
}
return $code;
}
sub _build_methods
{
my $me = shift;
my ($class, $attr, $spec) = @_;
my @code;
if ($spec->{is} eq q(rwp))
{
push @code,
$me->_build_reader($class, $attr, $spec, $attr),
$me->_build_writer($class, $attr, $spec, "_set_$attr");
}
elsif ($spec->{is} eq q(rw))
lib/Acme/Has/Tiny.pm view on Meta::CPAN
? sprintf('sub %s { %s; $_[0]{%s} = $_[1] }', $method, $inlined, perlstring($attr))
: sprintf('sub %s { $_[0]{%s} = $_[1] }', $method, perlstring($attr));
}
sub _build_accessor
{
my $me = shift;
my ($class, $attr, $spec, $method) = @_;
my $inlined;
my $isa = $spec->{isa};
if (blessed($isa) and $isa->can_be_inlined)
{
$inlined = $isa->inline_assert('$_[1]');
}
elsif ($isa)
{
$inlined = sprintf('$Acme::Has::Tiny::ATTRIBUTES{%s}{%s}{isa}->($_[1]);', perlstring($class), perlstring($attr));
}
if (CAN_HAZ_XS and not $inlined)
{
"Class::XSAccessor"->import(class => $class, accessors => { $method => $attr });
return;
}
return defined($inlined)
? sprintf('sub %s { return $_[0]{%s} unless @_; %s; $_[0]{%s} = $_[1] }', $method, perlstring($attr), $inlined, perlstring($attr))
: sprintf('sub %s { return $_[0]{%s} unless @_; $_[0]{%s} = $_[1] }', $method, perlstring($attr), perlstring($attr));
}
sub _compile_validator
{
my $me = shift;
my $code = join "\n" => (
"#line 1 \"validator(Acme::Has::Tiny)\"",
"package $_[0];",
'sub {',
'my $self = $_[0];',
$me->_build_validator_parts(@_),
'return $self;',
'}',
);
eval $code;
}
sub _build_validator_parts
{
my $me = shift;
my ($class, $attributes) = @_;
my @code;
for my $a (sort keys %$attributes)
{
my $spec = $attributes->{$a};
if ($spec->{default})
{
push @code, sprintf(
'exists($self->{%s}) or $self->{%s} = $Acme::Has::Tiny::ATTRIBUTES{%s}{%s}{default}->();',
map perlstring($_), $a, $a, $class, $a,
);
}
elsif ($spec->{required})
{
push @code, sprintf(
'exists($self->{%s}) or Acme::Has::Tiny::_croak("Attribute %%s is required by %%s", %s, %s);',
map perlstring($_), $a, $a, $class,
);
}
my $isa = $spec->{isa};
if (blessed($isa) and $isa->can_be_inlined)
{
push @code, (
sprintf('if (exists($self->{%s})) {', $a),
$isa->inline_assert(sprintf '$self->{%s}', perlstring($a)),
'}',
);
}
elsif ($isa)
{
push @code, (
sprintf('if (exists($self->{%s})) {', $a),
sprintf('$Acme::Has::Tiny::ATTRIBUTES{%s}{%s}{isa}->($self->{%s});', map perlstring($_), $class, $a, $a),
'}',
);
}
}
return @code;
}
sub _find_parents
{
my $me = shift;
my $class = $_[0];
if (eval { require mro } or eval { require MRO::Compat })
{
return @{ mro::get_linear_isa($class) };
}
require Class::ISA;
return Class::ISA::self_and_super_path($class);
}
1;
__END__
=pod
=encoding utf-8
=for stopwords ro rw rwp isa
=head1 NAME
Acme::Has::Tiny - tiny implementation of Moose-like "has" keyword
=head1 SYNOPSIS
package Person;
use Acme::Has::Tiny qw(new has);
use Types::Standard -types;
has name => (isa => Str);
has age => (isa => Num);
=head1 DESCRIPTION
Acme::Has::Tiny provides a Moose-like C<has> function. It is not
particularly full-featured, providing just enough to be useful for
small OO projects.
Generally speaking, I'd recommend using L<Moo> or L<Moose> instead, but
if you want to use this then I'm fairly unlikely to hunt you down with dogs.
This module was originally written for Type::Tiny, but turned out to be
just a smidgen slower than the system it was replacing, so was abandoned.
=head2 Methods
( run in 0.638 second using v1.01-cache-2.11-cpan-140bd7fdf52 )