Commit | Line | Data |
306290e8 |
1 | package Mouse::Meta::Attribute; |
bc69ee88 |
2 | use Mouse::Util qw(:meta); # enables strict and warnings |
c3398f5b |
3 | |
2efc0af1 |
4 | use Carp (); |
2efc0af1 |
5 | |
684db121 |
6 | use Mouse::Meta::TypeConstraint; |
d7d8d49b |
7 | |
6fd8aa82 |
8 | #my @attributes = qw( |
9 | # name init_arg required |
10 | # is |
11 | # isa does coerce |
12 | # accessor reader writer predicate clearer |
13 | # default builder lazy lazy_build |
14 | # auto_deref trigger |
15 | # metaclass traits |
16 | # |
17 | # handles |
18 | # documentation |
19 | # |
20 | # type_constraint |
21 | #); |
22 | |
87ca293b |
23 | sub _process_options{ |
24 | my($class, $name, $args) = @_; |
c3398f5b |
25 | |
2efc0af1 |
26 | # taken from Class::MOP::Attribute::new |
27 | |
28 | defined($name) |
29 | or $class->throw_error('You must provide a name for the attribute'); |
2e7e86c6 |
30 | |
1b9e472d |
31 | if(!exists $args->{init_arg}){ |
32 | $args->{init_arg} = $name; |
2efc0af1 |
33 | } |
34 | |
35 | # 'required' requires eigher 'init_arg', 'builder', or 'default' |
1b9e472d |
36 | my $can_be_required = defined( $args->{init_arg} ); |
2efc0af1 |
37 | |
1b9e472d |
38 | if(exists $args->{builder}){ |
c6d8f5ca |
39 | # XXX: |
40 | # Moose refuses a CODE ref builder, but Mouse doesn't for backward compatibility |
41 | # This feature will be changed in a future. (gfx) |
121acb8a |
42 | $class->throw_error('builder must be a defined scalar value which is a method name') |
c6d8f5ca |
43 | #if ref $args->{builder} || !defined $args->{builder}; |
44 | if !defined $args->{builder}; |
2efc0af1 |
45 | |
46 | $can_be_required++; |
47 | } |
1b9e472d |
48 | elsif(exists $args->{default}){ |
49 | if(ref $args->{default} && ref($args->{default}) ne 'CODE'){ |
2efc0af1 |
50 | $class->throw_error("References are not allowed as default values, you must " |
51 | . "wrap the default of '$name' in a CODE reference (ex: sub { [] } and not [])"); |
52 | } |
53 | $can_be_required++; |
54 | } |
55 | |
1b9e472d |
56 | if( $args->{required} && !$can_be_required ) { |
121acb8a |
57 | $class->throw_error("You cannot have a required attribute ($name) without a default, builder, or an init_arg"); |
2efc0af1 |
58 | } |
59 | |
1b9e472d |
60 | # taken from Mouse::Meta::Attribute->new and _process_args-> |
2efc0af1 |
61 | |
1b9e472d |
62 | if(exists $args->{is}){ |
63 | my $is = $args->{is}; |
2efc0af1 |
64 | |
65 | if($is eq 'ro'){ |
1b9e472d |
66 | $args->{reader} ||= $name; |
2efc0af1 |
67 | } |
68 | elsif($is eq 'rw'){ |
1b9e472d |
69 | if(exists $args->{writer}){ |
70 | $args->{reader} ||= $name; |
2efc0af1 |
71 | } |
72 | else{ |
1b9e472d |
73 | $args->{accessor} ||= $name; |
2efc0af1 |
74 | } |
75 | } |
76 | elsif($is eq 'bare'){ |
77 | # do nothing, but don't complain (later) about missing methods |
78 | } |
79 | else{ |
80 | $is = 'undef' if !defined $is; |
81 | $class->throw_error("I do not understand this option (is => $is) on attribute ($name)"); |
82 | } |
83 | } |
84 | |
85 | my $tc; |
1b9e472d |
86 | if(exists $args->{isa}){ |
87 | $args->{type_constraint} = Mouse::Util::TypeConstraints::find_or_create_isa_type_constraint($args->{isa}); |
2efc0af1 |
88 | } |
1b9e472d |
89 | elsif(exists $args->{does}){ |
9f74c401 |
90 | $args->{type_constraint} = Mouse::Util::TypeConstraints::find_or_create_does_type_constraint($args->{does}); |
2efc0af1 |
91 | } |
1b9e472d |
92 | $tc = $args->{type_constraint}; |
2efc0af1 |
93 | |
1b9e472d |
94 | if($args->{coerce}){ |
2efc0af1 |
95 | defined($tc) |
96 | || $class->throw_error("You cannot have coercion without specifying a type constraint on attribute ($name)"); |
97 | |
1b9e472d |
98 | $args->{weak_ref} |
121acb8a |
99 | && $class->throw_error("You cannot have a weak reference to a coerced value on attribute ($name)"); |
2efc0af1 |
100 | } |
101 | |
1b9e472d |
102 | if ($args->{lazy_build}) { |
103 | exists($args->{default}) |
121acb8a |
104 | && $class->throw_error("You can not use lazy_build and default for the same attribute ($name)"); |
105 | |
1b9e472d |
106 | $args->{lazy} = 1; |
107 | $args->{builder} ||= "_build_${name}"; |
121acb8a |
108 | if ($name =~ /^_/) { |
1b9e472d |
109 | $args->{clearer} ||= "_clear${name}"; |
110 | $args->{predicate} ||= "_has${name}"; |
121acb8a |
111 | } |
112 | else { |
1b9e472d |
113 | $args->{clearer} ||= "clear_${name}"; |
114 | $args->{predicate} ||= "has_${name}"; |
121acb8a |
115 | } |
2efc0af1 |
116 | } |
117 | |
1b9e472d |
118 | if ($args->{auto_deref}) { |
121acb8a |
119 | defined($tc) |
120 | || $class->throw_error("You cannot auto-dereference without specifying a type constraint on attribute ($name)"); |
2efc0af1 |
121 | |
121acb8a |
122 | ( $tc->is_a_type_of('ArrayRef') || $tc->is_a_type_of('HashRef') ) |
123 | || $class->throw_error("You cannot auto-dereference anything other than a ArrayRef or HashRef on attribute ($name)"); |
2efc0af1 |
124 | } |
125 | |
1b9e472d |
126 | if (exists $args->{trigger}) { |
127 | ('CODE' eq ref $args->{trigger}) |
121acb8a |
128 | || $class->throw_error("Trigger must be a CODE ref on attribute ($name)"); |
2efc0af1 |
129 | } |
45959ffa |
130 | |
1b9e472d |
131 | if ($args->{lazy}) { |
132 | (exists $args->{default} || defined $args->{builder}) |
121acb8a |
133 | || $class->throw_error("You cannot have lazy attribute ($name) without specifying a default value for it"); |
2efc0af1 |
134 | } |
90fe520e |
135 | |
87ca293b |
136 | return; |
137 | } |
138 | |
139 | sub new { |
140 | my $class = shift; |
141 | my $name = shift; |
142 | |
143 | my %args = (@_ == 1) ? %{ $_[0] } : @_; |
144 | |
145 | $class->_process_options($name, \%args); |
146 | |
147 | $args{name} = $name; |
148 | |
926290ac |
149 | my $self = bless \%args, $class; |
2efc0af1 |
150 | |
1b9e472d |
151 | # extra attributes |
152 | if($class ne __PACKAGE__){ |
926290ac |
153 | $class->meta->_initialize_object($self, \%args); |
90fe520e |
154 | } |
c3398f5b |
155 | |
2efc0af1 |
156 | # XXX: there is no fast way to check attribute validity |
1b9e472d |
157 | # my @bad = ...; |
2efc0af1 |
158 | # if(@bad){ |
159 | # @bad = sort @bad; |
160 | # Carp::cluck("Found unknown argument(s) passed to '$name' attribute constructor in '$class': @bad"); |
161 | # } |
162 | |
926290ac |
163 | return $self; |
c3398f5b |
164 | } |
165 | |
43165725 |
166 | sub has_read_method { $_[0]->has_reader || $_[0]->has_accessor } |
167 | sub has_write_method { $_[0]->has_writer || $_[0]->has_accessor } |
1b9e472d |
168 | |
87ca293b |
169 | sub interpolate_class{ |
8cf51b82 |
170 | my($class, $args) = @_; |
c3398f5b |
171 | |
1b9e472d |
172 | if(my $metaclass = delete $args->{metaclass}){ |
173 | $class = Mouse::Util::resolve_metaclass_alias( Attribute => $metaclass ); |
174 | } |
1bfebf5f |
175 | |
87ca293b |
176 | my @traits; |
1b9e472d |
177 | if(my $traits_ref = delete $args->{traits}){ |
87ca293b |
178 | |
f3f04eed |
179 | for (my $i = 0; $i < @{$traits_ref}; $i++) { |
180 | my $trait = Mouse::Util::resolve_metaclass_alias(Attribute => $traits_ref->[$i], trait => 1); |
b2500191 |
181 | |
f3f04eed |
182 | next if $class->does($trait); |
74be9f76 |
183 | |
f3f04eed |
184 | push @traits, $trait; |
1b9e472d |
185 | |
f3f04eed |
186 | # are there options? |
187 | push @traits, $traits_ref->[++$i] |
188 | if ref($traits_ref->[$i+1]); |
189 | } |
c3398f5b |
190 | |
1b9e472d |
191 | if (@traits) { |
192 | $class = Mouse::Meta::Class->create_anon_class( |
193 | superclasses => [ $class ], |
194 | roles => \@traits, |
195 | cache => 1, |
196 | )->name; |
1b9e472d |
197 | } |
74be9f76 |
198 | } |
199 | |
87ca293b |
200 | return( $class, @traits ); |
1b9e472d |
201 | } |
202 | |
ffbbf459 |
203 | sub _coerce_and_verify { |
204 | my($self, $value, $instance) = @_; |
205 | |
206 | my $type_constraint = $self->{type_constraint}; |
4331a3f9 |
207 | return $value if !defined $type_constraint; |
ffbbf459 |
208 | |
209 | if ($self->should_coerce && $type_constraint->has_coercion) { |
210 | $value = $type_constraint->coerce($value); |
211 | } |
212 | |
ffbbf459 |
213 | $self->verify_against_type_constraint($value); |
214 | |
215 | return $value; |
216 | } |
217 | |
20e25eb9 |
218 | sub verify_against_type_constraint { |
f55f60dd |
219 | my ($self, $value) = @_; |
5aa30ced |
220 | |
ffbbf459 |
221 | my $type_constraint = $self->{type_constraint}; |
4331a3f9 |
222 | return 1 if !$type_constraint; |
ffbbf459 |
223 | return 1 if $type_constraint->check($value); |
5aa30ced |
224 | |
da23cd4a |
225 | $self->_throw_type_constraint_error($value, $type_constraint); |
b3b74cc6 |
226 | } |
5aa30ced |
227 | |
da23cd4a |
228 | sub _throw_type_constraint_error { |
2a96ea85 |
229 | my($self, $value, $type) = @_; |
230 | |
231 | $self->throw_error( |
232 | sprintf q{Attribute (%s) does not pass the type constraint because: %s}, |
233 | $self->name, |
234 | $type->get_message($value), |
235 | ); |
5aa30ced |
236 | } |
237 | |
1b9e472d |
238 | sub clone_and_inherit_options{ |
8cf51b82 |
239 | my($self, %args) = @_; |
240 | |
241 | my($attribute_class, @traits) = ref($self)->interpolate_class(\%args); |
1b9e472d |
242 | |
8cf51b82 |
243 | $args{traits} = \@traits if @traits; |
4f41d79b |
244 | # do not inherit the 'handles' attribute |
245 | foreach my $name(keys %{$self}){ |
246 | if(!exists $args{$name} && $name ne 'handles'){ |
247 | $args{$name} = $self->{$name}; |
248 | } |
249 | } |
250 | return $attribute_class->new($self->name, %args); |
1b9e472d |
251 | } |
252 | |
0126c27c |
253 | sub get_read_method { |
751de1a1 |
254 | return $_[0]->reader || $_[0]->accessor |
df77fd72 |
255 | } |
0126c27c |
256 | sub get_write_method { |
751de1a1 |
257 | return $_[0]->writer || $_[0]->accessor |
258 | } |
259 | |
260 | sub _get_accessor_method_ref { |
261 | my($self, $type, $generator) = @_; |
262 | |
263 | my $metaclass = $self->associated_class |
264 | || $self->throw_error('No asocciated class for ' . $self->name); |
265 | |
266 | my $accessor = $self->$type(); |
267 | if($accessor){ |
268 | return $metaclass->get_method_body($accessor); |
269 | } |
270 | else{ |
271 | return $self->accessor_metaclass->$generator($self, $metaclass); |
272 | } |
df77fd72 |
273 | } |
2a464664 |
274 | |
275 | sub get_read_method_ref{ |
276 | my($self) = @_; |
751de1a1 |
277 | return $self->{_read_method_ref} ||= $self->_get_accessor_method_ref('get_read_method', '_generate_reader'); |
2a464664 |
278 | } |
279 | |
280 | sub get_write_method_ref{ |
281 | my($self) = @_; |
751de1a1 |
282 | return $self->{_write_method_ref} ||= $self->_get_accessor_method_ref('get_write_method', '_generate_writer'); |
283 | } |
2a464664 |
284 | |
751de1a1 |
285 | sub set_value { |
286 | my($self, $object, $value) = @_; |
287 | return $self->get_write_method_ref()->($object, $value); |
288 | } |
289 | |
290 | sub get_value { |
291 | my($self, $object) = @_; |
292 | return $self->get_read_method_ref()->($object); |
2a464664 |
293 | } |
294 | |
751de1a1 |
295 | sub has_value { |
296 | my($self, $object) = @_; |
060f9228 |
297 | my $accessor_ref = $self->{_predicate_ref} |
751de1a1 |
298 | ||= $self->_get_accessor_method_ref('predicate', '_generate_predicate'); |
299 | |
060f9228 |
300 | return $accessor_ref->($object); |
751de1a1 |
301 | } |
302 | |
303 | sub clear_value { |
304 | my($self, $object) = @_; |
060f9228 |
305 | my $accessor_ref = $self->{_crealer_ref} |
751de1a1 |
306 | ||= $self->_get_accessor_method_ref('clearer', '_generate_clearer'); |
307 | |
060f9228 |
308 | return $accessor_ref->($object); |
751de1a1 |
309 | } |
310 | |
311 | |
04493075 |
312 | sub associate_method{ |
e5e22afd |
313 | my ($attribute, $method_name) = @_; |
04493075 |
314 | $attribute->{associated_methods}++; |
315 | return; |
316 | } |
317 | |
1b9e472d |
318 | sub install_accessors{ |
319 | my($attribute) = @_; |
320 | |
93540011 |
321 | my $metaclass = $attribute->associated_class; |
4ab51fb0 |
322 | my $accessor_class = $attribute->accessor_metaclass; |
1b9e472d |
323 | |
4ab51fb0 |
324 | foreach my $type(qw(accessor reader writer predicate clearer)){ |
1b9e472d |
325 | if(exists $attribute->{$type}){ |
4ab51fb0 |
326 | my $generator = '_generate_' . $type; |
327 | my $code = $accessor_class->$generator($attribute, $metaclass); |
328 | $metaclass->add_method($attribute->{$type} => $code); |
e5e22afd |
329 | $attribute->associate_method($attribute->{$type}); |
4ab51fb0 |
330 | } |
331 | } |
7ca5c5fb |
332 | |
4ab51fb0 |
333 | # install delegation |
334 | if(exists $attribute->{handles}){ |
335 | my %handles = $attribute->_canonicalize_handles($attribute->{handles}); |
feaa7084 |
336 | |
cbb81058 |
337 | while(my($handle, $method_to_call) = each %handles){ |
338 | $metaclass->add_method($handle => |
339 | $attribute->_make_delegation_method( |
340 | $handle, $method_to_call)); |
4ab51fb0 |
341 | |
cbb81058 |
342 | $attribute->associate_method($handle); |
1b9e472d |
343 | } |
344 | } |
345 | |
1b9e472d |
346 | return; |
347 | } |
348 | |
cbb81058 |
349 | sub delegation_metaclass() { 'Mouse::Meta::Method::Delegation' } |
350 | |
351 | sub _canonicalize_handles { |
352 | my($self, $handles) = @_; |
353 | |
354 | if (ref($handles) eq 'HASH') { |
355 | return %$handles; |
356 | } |
357 | elsif (ref($handles) eq 'ARRAY') { |
358 | return map { $_ => $_ } @$handles; |
359 | } |
360 | elsif (ref($handles) eq 'Regexp') { |
361 | my $class_or_role = ($self->{isa} || $self->{does}) |
362 | || $self->throw_error("Cannot delegate methods based on a Regexp without a type constraint (isa)"); |
363 | |
364 | my $meta = Mouse::Meta::Class->initialize("$class_or_role"); # "" for stringify |
365 | return map { $_ => $_ } |
366 | grep { !Mouse::Object->can($_) && $_ =~ $handles } |
367 | Mouse::Util::is_a_metarole($meta) |
368 | ? $meta->get_method_list |
369 | : $meta->get_all_method_names; |
370 | } |
371 | else { |
372 | $self->throw_error("Unable to canonicalize the 'handles' option with $handles"); |
373 | } |
374 | } |
375 | |
376 | sub _make_delegation_method { |
377 | my($self, $handle, $method_to_call) = @_; |
378 | my $delegator = $self->delegation_metaclass; |
379 | Mouse::Util::load_class($delegator); |
380 | |
381 | return $delegator->_generate_delegation($self, $handle, $method_to_call); |
382 | } |
383 | |
fce211ae |
384 | sub throw_error{ |
385 | my $self = shift; |
386 | |
387 | my $metaclass = (ref $self && $self->associated_class) || 'Mouse::Meta::Class'; |
388 | $metaclass->throw_error(@_, depth => 1); |
1bfebf5f |
389 | } |
390 | |
c3398f5b |
391 | 1; |
c3398f5b |
392 | __END__ |
393 | |
394 | =head1 NAME |
395 | |
bedd575c |
396 | Mouse::Meta::Attribute - The Mouse attribute metaclass |
c3398f5b |
397 | |
a25ca8d6 |
398 | =head1 VERSION |
399 | |
81fd550d |
400 | This document describes Mouse version 0.45 |
a25ca8d6 |
401 | |
c3398f5b |
402 | =head1 METHODS |
403 | |
1820fffe |
404 | =head2 C<< new(%options) -> Mouse::Meta::Attribute >> |
c3398f5b |
405 | |
306290e8 |
406 | Instantiates a new Mouse::Meta::Attribute. Does nothing else. |
c3398f5b |
407 | |
1820fffe |
408 | It adds the following options to the constructor: |
c3398f5b |
409 | |
1820fffe |
410 | =over 4 |
c3398f5b |
411 | |
612d3e1a |
412 | =item C<< is => 'ro', 'rw', 'bare' >> |
c3398f5b |
413 | |
1820fffe |
414 | This provides a shorthand for specifying the C<reader>, C<writer>, or |
415 | C<accessor> names. If the attribute is read-only ('ro') then it will |
416 | have a C<reader> method with the same attribute as the name. |
c3398f5b |
417 | |
1820fffe |
418 | If it is read-write ('rw') then it will have an C<accessor> method |
419 | with the same name. If you provide an explicit C<writer> for a |
420 | read-write attribute, then you will have a C<reader> with the same |
421 | name as the attribute, and a C<writer> with the name you provided. |
c3398f5b |
422 | |
1820fffe |
423 | Use 'bare' when you are deliberately not installing any methods |
424 | (accessor, reader, etc.) associated with this attribute; otherwise, |
425 | Moose will issue a deprecation warning when this attribute is added to a |
426 | metaclass. |
c3398f5b |
427 | |
612d3e1a |
428 | =item C<< isa => Type >> |
ab27a55e |
429 | |
1820fffe |
430 | This option accepts a type. The type can be a string, which should be |
431 | a type name. If the type name is unknown, it is assumed to be a class |
432 | name. |
ab27a55e |
433 | |
1820fffe |
434 | This option can also accept a L<Moose::Meta::TypeConstraint> object. |
ab27a55e |
435 | |
1820fffe |
436 | If you I<also> provide a C<does> option, then your C<isa> option must |
437 | be a class name, and that class must do the role specified with |
438 | C<does>. |
ab27a55e |
439 | |
612d3e1a |
440 | =item C<< does => Role >> |
ab27a55e |
441 | |
1820fffe |
442 | This is short-hand for saying that the attribute's type must be an |
443 | object which does the named role. |
c3398f5b |
444 | |
1820fffe |
445 | B<This option is not yet supported.> |
c3398f5b |
446 | |
612d3e1a |
447 | =item C<< coerce => Bool >> |
ab27a55e |
448 | |
1820fffe |
449 | This option is only valid for objects with a type constraint |
450 | (C<isa>). If this is true, then coercions will be applied whenever |
451 | this attribute is set. |
c3398f5b |
452 | |
1820fffe |
453 | You can make both this and the C<weak_ref> option true. |
c3398f5b |
454 | |
612d3e1a |
455 | =item C<< trigger => CodeRef >> |
ab27a55e |
456 | |
1820fffe |
457 | This option accepts a subroutine reference, which will be called after |
458 | the attribute is set. |
ab27a55e |
459 | |
612d3e1a |
460 | =item C<< required => Bool >> |
ab27a55e |
461 | |
1820fffe |
462 | An attribute which is required must be provided to the constructor. An |
463 | attribute which is required can also have a C<default> or C<builder>, |
464 | which will satisfy its required-ness. |
ab27a55e |
465 | |
1820fffe |
466 | A required attribute must have a C<default>, C<builder> or a |
467 | non-C<undef> C<init_arg> |
ab27a55e |
468 | |
612d3e1a |
469 | =item C<< lazy => Bool >> |
ab27a55e |
470 | |
1820fffe |
471 | A lazy attribute must have a C<default> or C<builder>. When an |
472 | attribute is lazy, the default value will not be calculated until the |
473 | attribute is read. |
93f08899 |
474 | |
612d3e1a |
475 | =item C<< weak_ref => Bool >> |
0fff36e6 |
476 | |
1820fffe |
477 | If this is true, the attribute's value will be stored as a weak |
478 | reference. |
c3398f5b |
479 | |
612d3e1a |
480 | =item C<< auto_deref => Bool >> |
fb706f5c |
481 | |
1820fffe |
482 | If this is true, then the reader will dereference the value when it is |
483 | called. The attribute must have a type constraint which defines the |
484 | attribute as an array or hash reference. |
485 | |
612d3e1a |
486 | =item C<< lazy_build => Bool >> |
1820fffe |
487 | |
488 | Setting this to true makes the attribute lazy and provides a number of |
489 | default methods. |
fb706f5c |
490 | |
1820fffe |
491 | has 'size' => ( |
492 | is => 'ro', |
493 | lazy_build => 1, |
494 | ); |
93d190e0 |
495 | |
1820fffe |
496 | is equivalent to this: |
93d190e0 |
497 | |
1820fffe |
498 | has 'size' => ( |
499 | is => 'ro', |
500 | lazy => 1, |
501 | builder => '_build_size', |
502 | clearer => 'clear_size', |
503 | predicate => 'has_size', |
504 | ); |
93d190e0 |
505 | |
1820fffe |
506 | =back |
507 | |
e5e22afd |
508 | =head2 C<< associate_method(MethodName) >> |
31c5194b |
509 | |
510 | Associates a method with the attribute. Typically, this is called internally |
511 | when an attribute generates its accessors. |
512 | |
e5e22afd |
513 | Currently the argument I<MethodName> is ignored in Mouse. |
31c5194b |
514 | |
1820fffe |
515 | =head2 C<< verify_against_type_constraint(Item) -> TRUE | ERROR >> |
516 | |
517 | Checks that the given value passes this attribute's type constraint. Returns C<true> |
518 | on success, otherwise C<confess>es. |
93d190e0 |
519 | |
1820fffe |
520 | =head2 C<< clone_and_inherit_options(options) -> Mouse::Meta::Attribute >> |
f7b11a21 |
521 | |
612d3e1a |
522 | Creates a new attribute in the owner class, inheriting options from parent classes. |
f7b11a21 |
523 | Accessors and helper methods are installed. Some error checking is done. |
524 | |
9ae9702e |
525 | =head2 C<< get_read_method_ref >> |
526 | |
527 | =head2 C<< get_write_method_ref >> |
528 | |
529 | Returns the subroutine reference of a method suitable for reading or |
530 | writing the attribute's value in the associated class. These methods |
531 | always return a subroutine reference, regardless of whether or not the |
df77fd72 |
532 | attribute is read- or write-only. |
533 | |
1820fffe |
534 | =head1 SEE ALSO |
f7b11a21 |
535 | |
1820fffe |
536 | L<Moose::Meta::Attribute> |
f7b11a21 |
537 | |
31c5194b |
538 | L<Class::MOP::Attribute> |
539 | |
c3398f5b |
540 | =cut |
541 | |