Moose::Cookbook::Recipe21 - The meta-attribute example |
Moose::Cookbook::Recipe21 - The meta-attribute example
package MyApp::Meta::Attribute::Labeled; use Moose; extends 'Moose::Meta::Attribute';
has label => ( is => 'rw', isa => 'Str', predicate => 'has_label', );
package Moose::Meta::Attribute::Custom::Labeled; sub register_implementation { 'MyApp::Meta::Attribute::Labeled' }
package MyApp::Website; use Moose; use MyApp::Meta::Attribute::Labeled;
has url => ( metaclass => 'Labeled', isa => 'Str', is => 'rw', label => "The site's URL", );
has name => ( is => 'rw', isa => 'Str', );
sub dump { my $self = shift;
# iterate over all the attributes in $self my %attributes = %{ $self->meta->get_attribute_map }; while (my ($name, $attribute) = each %attributes) {
# print the label if available if ($attribute->isa('MyApp::Meta::Attribute::Labeled') && $attribute->has_label) { print $attribute->label; } # otherwise print the name else { print $name; }
# print the attribute's value my $reader = $attribute->get_read_method; print ": " . $self->$reader . "\n"; } }
package main; my $app = MyApp::Website->new(url => "http://google.com", name => "Google"); $app->dump;
In this recipe, we begin to really delve into the wonder of meta-programming. Some readers may scoff and claim that this is the arena only of the most twisted Moose developers. Absolutely not! Any sufficiently twisted developer can benefit greatly from going more meta.
The high-level goal of this recipe's code is to allow each attribute to have a human-readable ``label'' attached to it. Such labels would be used when showing data to an end user. In this recipe we label the ``url'' attribute with ``The site's URL'' and create a simple method to demonstrate how to use that label.
All the attributes of a Moose-based object are actually objects themselves. These objects have methods and (surprisingly) attributes. Let's look at a concrete example.
has 'x' => (isa => 'Int', is => 'ro'); has 'y' => (isa => 'Int', is => 'rw');
Ahh, the veritable x and y of the Point example. Internally, every Point has an
x object and a y object. They have methods (such as ``get_value'') and attributes
(such as ``is_lazy''). What class are they instances of?
the Moose::Meta::Attribute manpage. You don't normally see the objects lurking behind
the scenes, because you usually just use $point->x
and $point->y
and forget that there's a lot of machinery lying in such methods.
So you have a $point
object, which has x
and y
methods. How can you
actually access the objects behind these attributes? Here's one way:
$point->meta->get_attribute_map()
get_attribute_map
returns a hash reference that maps attribute names to
their objects. In our case, get_attribute_map
might return something that
looks like the following:
{ x => Moose::Meta::Attribute=HASH(0x196c23c), y => Moose::Meta::Attribute=HASH(0x18d1690), }
Another way to get a handle on an attribute's object is
$self->meta->get_attribute('name')
. Here's one thing you can do now that
you can interact with the attribute's object directly:
print $point->meta->get_attribute('x')->type_constraint; => Int
(As an aside, it's not called ->isa
because $obj->isa
is already
taken)
So to actually beef up attributes, what we need to do is:
Moose makes both of these easy!
Let's start dissecting the recipe's code.
We get the ball rolling by creating a new attribute metaclass. It starts off somewhat ungloriously.
package MyApp::Meta::Attribute::Labeled; use Moose; extends 'Moose::Meta::Attribute';
You subclass metaclasses the same way you subclass regular classes. (Extra credit: how in the actual hell can you use the MOP to extend itself?)
has label => ( is => 'rw', isa => 'Str', predicate => 'has_label', );
Hey, this looks pretty reasonable! This is plain jane Moose code. Recipe 1 fare. This is merely making a new attribute. An attribute that attributes have. A meta-attribute. It may sound scary, but it really isn't! Reread REAL ATTRIBUTES 101 if this really is terrifying.
The name is ``label'', it will have a regular accessor, and is a string.
predicate
is a standard part of has
. It just creates a method that asks
the question ``Does this attribute have a value?''
package Moose::Meta::Attribute::Custom::Labeled; sub register_implementation { 'MyApp::Meta::Attribute::Labeled' }
This lets Moose discover our new metaclass. That way attributes can actually use it. More on what this is doing in a moment.
Note that we're done defining the new metaclass! Only nine lines of code, and not particularly difficult lines, either. Now to start using the metaclass.
package MyApp::Website; use Moose; use MyApp::Meta::Attribute::Labeled;
Nothing new here. We do have to actually load our metaclass to be able to use it.
has url => ( metaclass => 'Labeled', isa => 'Str', is => 'rw', label => "The site's URL", );
Ah ha! Now we're using the metaclass. We're adding a new attribute, url
, to
MyApp::Website
. has
lets you set the metaclass of the attribute.
Ordinarily (as we've seen), the metaclass is Moose::Meta::Attribute
.
When has
sees that you're using a new metaclass, it will take the
metaclass's name, prepend Moose::Meta::Attribute::Custom::
, and call the
register_implementation
function in that package. So here Moose calls
Moose::Meta::Attribute::Custom::Labeled::register_implementation
. We defined
that function in the beginning -- it just returns our ``real'' metaclass'
package, MyApp::Meta::Attribute::Labeled
. So Moose uses that metaclass for
the attribute. It may seem a bit convoluted, but the alternative would be to
use metaclass => 'MyApp::Meta::Attribute::Labeled'
on every attribute.
As usual, Moose optimizes in favor of the end user, not the metaprogrammer. :)
We also could have just defined the metaclass in
Moose::Meta::Attribute::Custom::Labeled
, but it's probably better to keep to
your own namespaces.
Finally, we see that has
is setting our new meta-attribute, label
, to
"The site's URL"
. We can access this meta-attribute with:
$website->meta->get_attribute('url')->label()
Well, back to the code.
has name => ( is => 'rw', isa => 'Str', );
Of course, you don't have to use the new metaclass for all new attributes.
Now we begin defining a method that will dump the MyApp::Website
instance
for human readers.
sub dump { my $self = shift;
# iterate over all the attributes in $self my %attributes = %{ $self->meta->get_attribute_map }; while (my ($name, $attribute) = each %attributes) {
Recall that get_attribute_map
returns a hashref of attribute names and their
associated objects.
# print the label if available if ($attribute->isa('MyApp::Meta::Attribute::Labeled') && $attribute->has_label) { print $attribute->label; }
We have two checks here. The first is ``is this attribute an instance of
MyApp::Meta::Attribute::Labeled
?''. It's good to code defensively. Even if
all of your attributes have this metaclass, you never know when someone is
going to subclass your work of art. Poorly. In other words, it's likely that
there will still be (many) attributes that are instances of the default
Moose::Meta::Attribute
.
The second check is ``does this attribute have a label?''. This method was defined in the new metaclass as the ``predicate''. If we pass both checks, we print the attribute's label.
# otherwise print the name else { print $name; }
Another good, defensive coding practice: Provide reasonable defaults.
# print the attribute's value my $reader = $attribute->get_read_method; print ": " . $self->$reader . "\n"; } }
Here's another example of using the attribute metaclass.
$attribute->get_read_method
returns the name of the method that can
be invoked on the original object to read the attribute's value.
$self->$reader
is an example of ``reflection'' -- instead of using the
name of the method, we're using a variable with the name of the method in it.
Perl doesn't mind. Another way to write this would be
$self->can($reader)->($self)
. Yuck. :)
package main; my $app = MyApp::Website->new(url => "http://google.com", name => "Google"); $app->dump;
And we wrap up the example with a script to show off our newfound magic.
Why oh why would you want to go through all of these contortions when you can
just print ``The site's URL'' directly in the dump
method? For one, the DRY
(Don't Repeat Yourself) principle. If you have it in the dump
method, you'll
probably also have it in the as_form
method, and to_file
, and so on. So
why not have a method that maps attribute names to labels? That could work, but
why not include the label where it belongs, in the attribute's definition?
That way you're also less likely to forget to add the label.
More importantly, this was a very simple example. Your metaclasses aren't limited to just adding new meta-attributes. For example, you could implement a metaclass that expires attributes after a certain amount of time. You might use it as such:
has site_cache => ( metaclass => 'TimedExpiry', expires_after => { hours => 1 }, refresh_with => sub { get($_->url) }, isa => 'Str', is => 'ro', );
The sky's the limit!
Shawn M Moore <sartak@gmail.com>
Copyright 2006-2008 by Infinity Interactive, Inc.
This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself.
Moose::Cookbook::Recipe21 - The meta-attribute example |