my $type_constraint_obj = $attr->type_constraint;
my $environment = {
'$attr' => \$attr,
- '$attr_name' => \$attr->name,
'$meta' => \$self,
'$type_constraint_obj' => \$type_constraint_obj,
- '$type_constraint_name' => \($type_constraint_obj && $type_constraint_obj->name),
'$type_constraint' => \($type_constraint_obj
? $type_constraint_obj->_compiled_type_constraint
: undef),
sub _generate_accessor_method_inline {
my $self = $_[0];
my $attr = $self->associated_attribute;
- my $attr_name = $attr->name;
my $inv = '$_[0]';
my $value_name = $self->_value_needs_copy ? '$val' : '$_[1]';
sub _generate_writer_method_inline {
my $self = $_[0];
my $attr = $self->associated_attribute;
- my $attr_name = $attr->name;
my $inv = '$_[0]';
- my $slot_access = $self->_inline_get($inv, $attr_name);
+ my $slot_access = $self->_inline_get($inv);
my $value_name = $self->_value_needs_copy ? '$val' : '$_[1]';
$self->_eval_code('sub { '
sub _generate_reader_method_inline {
my $self = $_[0];
my $attr = $self->associated_attribute;
- my $attr_name = $attr->name;
my $inv = '$_[0]';
- my $slot_access = $self->_inline_get($inv, $attr_name);
+ my $slot_access = $self->_inline_get($inv);
$self->_eval_code('sub {'
. $self->_inline_pre_body(@_)
my ($self, $value) = @_;
my $attr = $self->associated_attribute;
- my $attr_name = $attr->name;
return '' unless $attr->has_type_constraint;
- my $type_constraint_name = $attr->type_constraint->name;
+ my $attr_name = quotemeta( $attr->name );
qq{\$type_constraint->($value) || } . $self->_inline_throw_error(qq{"Attribute ($attr_name) does not pass the type constraint because: " . \$type_constraint_obj->get_message($value)}, "data => $value") . ";";
}
my $self = shift;
my $attr = $self->associated_attribute;
- my $attr_name = $attr->name;
-
return '' unless $attr->is_required;
+
+ my $attr_name = quotemeta( $attr->name );
+
return qq{(\@_ >= 2) || } . $self->_inline_throw_error(qq{"Attribute ($attr_name) is required, so cannot be set to undef"}) . ';' # defined $_[1] is not good enough
}
return '' unless $attr->is_lazy;
- my $slot_exists = $self->_inline_has($instance, $attr->name);
+ my $slot_exists = $self->_inline_has($instance);
my $code = 'unless (' . $slot_exists . ') {' . "\n";
if ($attr->has_type_constraint) {
$sigil = '%';
}
else {
- $self->throw_error("Can not auto de-reference the type constraint '" . $type_constraint->name . "'", type_constraint => $type_constraint );
+ $self->throw_error( "Can not auto de-reference the type constraint '"
+ . quotemeta( $type_constraint->name )
+ . "'", type_constraint => $type_constraint );
}
"(wantarray() ? $sigil\{ ( $ref_value ) || return } : ( $ref_value ) )";
if ($is_moose && defined($attr->init_arg) && $attr->is_required && !$attr->has_default && !$attr->has_builder) {
push @source => ('(exists $params->{\'' . $attr->init_arg . '\'}) ' .
- '|| ' . $self->_inline_throw_error('"Attribute (' . $attr->name . ') is required"') .';');
+ '|| ' . $self->_inline_throw_error('"Attribute (' . quotemeta($attr->name) . ') is required"') .';');
}
if (($attr->has_default || $attr->has_builder) && !($is_moose && $attr->is_lazy)) {
my ($self, $attr, $type_constraint_cv, $type_constraint_obj, $value_name) = @_;
return (
$self->_inline_throw_error('"Attribute (' # FIXME add 'dad'
- . $attr->name
+ . quotemeta( $attr->name )
. ') does not pass the type constraint because: " . '
. $type_constraint_obj . '->get_message(' . $value_name . ')')
. "\n\t unless " . $type_constraint_cv . '->(' . $value_name . ');'
use warnings;
use Test::More;
+use Test::Moose;
{
package Foo;
default => 1,
);
+ # Assigning types to these non-alpha attrs exposed a bug in Moose.
has '@type' => (
+ isa => 'Str',
required => 0,
reader => 'get_at_type',
- default => 2,
+ writer => 'set_at_type',
+ default => 'at type',
);
has 'has spaces' => (
+ isa => 'Int',
required => 0,
reader => 'get_hs',
default => 42,
);
+ has '!req' => (
+ required => 1,
+ reader => 'req'
+ );
+
no Moose;
}
-{
- my $foo = Foo->new;
-
+with_immutable {
ok( Foo->meta->has_attribute($_), "Foo has '$_' attribute" )
for 'type', '@type', 'has spaces';
- is( $foo->get_type, 1, q{'type' attribute default is 1} );
- is( $foo->get_at_type, 2, q{'@type' attribute default is 1} );
- is( $foo->get_hs, 42, q{'has spaces' attribute default is 42} );
+ my $foo = Foo->new( '!req' => 42 );
+
+ is( $foo->get_type, 1, q{'type' attribute default is 1} );
+ is( $foo->get_at_type, 'at type', q{'@type' attribute default is 1} );
+ is( $foo->get_hs, 42, q{'has spaces' attribute default is 42} );
+
+ $foo = Foo->new(
+ type => 'foo',
+ '@type' => 'bar',
+ 'has spaces' => 200,
+ '!req' => 84,
+ );
+
+ isa_ok( $foo, 'Foo' );
+ is( $foo->get_at_type, 'bar', q{reader for '@type'} );
+ is( $foo->get_hs, 200, q{reader for 'has spaces'} );
- Foo->meta->make_immutable, redo if Foo->meta->is_mutable;
+ $foo->set_at_type(99);
+ is( $foo->get_at_type, 99, q{writer for '@type' worked} );
}
+'Foo';
done_testing;