X-Git-Url: http://git.shadowcat.co.uk/gitweb/gitweb.cgi?a=blobdiff_plain;f=lib%2FMoose.pm;h=9b8c5ebb435352f451f9bcace76c20a5f5406dfc;hb=52c7c33051150fe0851c98369b6c4cb25a1779c1;hp=7756cb48f5df325bf52fcfe800d76d4dd7fa6245;hpb=4276ccb4d121b35124dfc060b78cd91975a7e5ef;p=gitmo%2FMoose.git diff --git a/lib/Moose.pm b/lib/Moose.pm index 7756cb4..9b8c5eb 100644 --- a/lib/Moose.pm +++ b/lib/Moose.pm @@ -4,13 +4,14 @@ package Moose; use strict; use warnings; -our $VERSION = '0.09_03'; +our $VERSION = '0.21'; +our $AUTHORITY = 'cpan:STEVAN'; use Scalar::Util 'blessed', 'reftype'; use Carp 'confess'; use Sub::Name 'subname'; +use B 'svref_2object'; -use UNIVERSAL::require; use Sub::Exporter; use Class::MOP; @@ -34,6 +35,7 @@ use Moose::Util::TypeConstraints; subtype $class => as 'Object' => where { $_->isa($class) } + => optimize_as { blessed($_[0]) && $_[0]->isa($class) } unless find_type_constraint($class); my $meta; @@ -44,7 +46,7 @@ use Moose::Util::TypeConstraints; # override a specific class $meta = $class->meta(); (blessed($meta) && $meta->isa('Moose::Meta::Class')) - || confess "Whoops, not møøsey enough"; + || confess "You already have a &meta function, but it does not return a Moose::Meta::Class"; } else { # NOTE: @@ -68,91 +70,31 @@ use Moose::Util::TypeConstraints; my %exports = ( extends => sub { my $class = $CALLER; - return subname 'Moose::extends' => sub ($;@) { - _load_all_classes(@_); - my $meta = $class->meta; - foreach my $super (@_) { - # don't bother if it does not have a meta. - next unless $super->can('meta'); - # if it's meta is a vanilla Moose, - # then we can safely ignore it. - next if blessed($super->meta) eq 'Moose::Meta::Class'; - # but if we have anything else, - # we need to check it out ... - unless (# see if of our metaclass is incompatible - ($meta->isa(blessed($super->meta)) && - # and see if our instance metaclass is incompatible - $meta->instance_metaclass->isa($super->meta->instance_metaclass)) && - # ... and if we are just a vanilla Moose - $meta->isa('Moose::Meta::Class')) { - # re-initialize the meta ... - my $super_meta = $super->meta; - # NOTE: - # We might want to consider actually - # transfering any attributes from the - # original meta into this one, but in - # general you should not have any there - # at this point anyway, so it's very - # much an obscure edge case anyway - $meta = $super_meta->reinitialize($class => ( - ':attribute_metaclass' => $super_meta->attribute_metaclass, - ':method_metaclass' => $super_meta->method_metaclass, - ':instance_metaclass' => $super_meta->instance_metaclass, - )); - } - } + return subname 'Moose::extends' => sub (@) { + confess "Must derive at least one class" unless @_; + Class::MOP::load_class($_) for @_; + # this checks the metaclass to make sure + # it is correct, sometimes it can get out + # of sync when the classes are being built + my $meta = $class->meta->_fix_metaclass_incompatability(@_); $meta->superclasses(@_); }; }, with => sub { my $class = $CALLER; - return subname 'Moose::with' => sub ($;@) { + return subname 'Moose::with' => sub (@) { my (@roles) = @_; - _load_all_classes(@roles); - ($_->can('meta') && $_->meta->isa('Moose::Meta::Role')) - || confess "You can only consume roles, $_ is not a Moose role" - foreach @roles; - if (scalar @roles == 1) { - $roles[0]->meta->apply($class->meta); - } - else { - Moose::Meta::Role->combine( - map { $_->meta } @roles - )->apply($class->meta); - } + confess "Must specify at least one role" unless @roles; + Class::MOP::load_class($_) for @roles; + $class->meta->_apply_all_roles(@roles); }; }, has => sub { my $class = $CALLER; return subname 'Moose::has' => sub ($;%) { - my ($name, %options) = @_; - my $meta = $class->meta; - if ($name =~ /^\+(.*)/) { - my $inherited_attr = $meta->find_attribute_by_name($1); - (defined $inherited_attr) - || confess "Could not find an attribute by the name of '$1' to inherit from"; - my $new_attr; - if ($inherited_attr->isa('Moose::Meta::Attribute')) { - $new_attr = $inherited_attr->clone_and_inherit_options(%options); - } - else { - # NOTE: - # kind of a kludge to handle Class::MOP::Attributes - $new_attr = Moose::Meta::Attribute::clone_and_inherit_options( - $inherited_attr, %options - ); - } - $meta->add_attribute($new_attr); - } - else { - if ($options{metaclass}) { - _load_all_classes($options{metaclass}); - $meta->add_attribute($options{metaclass}->new($name, %options)); - } - else { - $meta->add_attribute($name, %options); - } - } + my ($name, %options) = @_; + my $attrs = (ref($name) eq 'ARRAY') ? $name : [($name)]; + $class->meta->_process_attribute($_, %options) for @$attrs; }; }, before => sub { @@ -180,6 +122,11 @@ use Moose::Util::TypeConstraints; }; }, super => sub { + { + our %SUPER_SLOT; + no strict 'refs'; + $SUPER_SLOT{$CALLER} = \*{"${CALLER}::super"}; + } return subname 'Moose::super' => sub {}; }, override => sub { @@ -190,6 +137,11 @@ use Moose::Util::TypeConstraints; }; }, inner => sub { + { + our %INNER_SLOT; + no strict 'refs'; + $INNER_SLOT{$CALLER} = \*{"${CALLER}::inner"}; + } return subname 'Moose::inner' => sub {}; }, augment => sub { @@ -199,12 +151,38 @@ use Moose::Util::TypeConstraints; $class->meta->add_augment_method_modifier($name => $method); }; }, + + # NOTE: + # this is experimental, but I am not + # happy with it. If you want to try + # it, you will have to uncomment it + # yourself. + # There is a really good chance that + # this will be deprecated, dont get + # too attached + # self => sub { + # return subname 'Moose::self' => sub {}; + # }, + # method => sub { + # my $class = $CALLER; + # return subname 'Moose::method' => sub { + # my ($name, $method) = @_; + # $class->meta->add_method($name, sub { + # my $self = shift; + # no strict 'refs'; + # no warnings 'redefine'; + # local *{$class->meta->name . '::self'} = sub { $self }; + # $method->(@_); + # }); + # }; + # }, + confess => sub { return \&Carp::confess; }, blessed => sub { return \&Scalar::Util::blessed; - } + }, ); my $exporter = Sub::Exporter::build_exporter({ @@ -227,31 +205,50 @@ use Moose::Util::TypeConstraints; goto $exporter; } + + sub unimport { + no strict 'refs'; + my $class = caller(); + # loop through the exports ... + foreach my $name (keys %exports) { + + # if we find one ... + if (defined &{$class . '::' . $name}) { + my $keyword = \&{$class . '::' . $name}; + + # make sure it is from Moose + my $pkg_name = eval { svref_2object($keyword)->GV->STASH->NAME }; + next if $@; + next if $pkg_name ne 'Moose'; + + # and if it is from Moose then undef the slot + delete ${$class . '::'}{$name}; + } + } + } + + } -## Utility functions - -sub _load_all_classes { - foreach my $super (@_) { - # see if this is already - # loaded in the symbol table - next if _is_class_already_loaded($super); - # otherwise require it ... - ($super->require) - || confess "Could not load superclass '$super' because : " . $UNIVERSAL::require::ERROR; - } -} +## make 'em all immutable -sub _is_class_already_loaded { - my $name = shift; - no strict 'refs'; - return 1 if defined ${"${name}::VERSION"} || defined @{"${name}::ISA"}; - foreach (keys %{"${name}::"}) { - next if substr($_, -2, 2) eq '::'; - return 1 if defined &{"${name}::$_"}; - } - return 0; -} +$_->meta->make_immutable( + inline_constructor => 0, + inline_accessors => 0, +) for ( + 'Moose::Meta::Attribute', + 'Moose::Meta::Class', + 'Moose::Meta::Instance', + + 'Moose::Meta::TypeConstraint', + 'Moose::Meta::TypeConstraint::Union', + 'Moose::Meta::TypeCoercion', + + 'Moose::Meta::Method', + 'Moose::Meta::Method::Accessor', + 'Moose::Meta::Method::Constructor', + 'Moose::Meta::Method::Overriden', +); 1; @@ -261,7 +258,7 @@ __END__ =head1 NAME -Moose - Moose, it's the new Camel +Moose - A complete modern object system for Perl 5 =head1 SYNOPSIS @@ -291,21 +288,7 @@ Moose - Moose, it's the new Camel after 'clear' => sub { my $self = shift; $self->z(0); - }; - -=head1 CAVEAT - -Moose is a rapidly maturing module, and is already being used by -a number of people. It's test suite is growing larger by the day, -and the docs should soon follow. - -This said, Moose is not yet finished, and should still be considered -to be evolving. Much of the outer API is stable, but the internals -are still subject to change (although not without serious thought -given to it). - -For more details, please refer to the L section of -this document. + }; =head1 DESCRIPTION @@ -314,7 +297,7 @@ Moose is an extension of the Perl 5 object system. =head2 Another object system!?!? Yes, I know there has been an explosion recently of new ways to -build object's in Perl 5, most of them based on inside-out objects, +build object's in Perl 5, most of them based on inside-out objects and other such things. Moose is different because it is not a new object system for Perl 5, but instead an extension of the existing object system. @@ -324,39 +307,50 @@ for Perl 5. This means that Moose not only makes building normal Perl 5 objects better, but it also provides the power of metaclass programming. -=head2 Can I use this in production? Or is this just an experiment? +=head2 Is this for real? Or is this just an experiment? Moose is I on the prototypes and experiments I did for the Perl 6 -meta-model, however Moose is B an experiment/prototype, it is -for B. I will be deploying Moose into production environments later -this year, and I have all intentions of using it as my de-facto class -builderfrom now on. +meta-model; however Moose is B an experiment/prototype, it is +for B. + +=head2 Is this ready for use in production? + +Yes, I believe that it is. + +I have two medium-to-large-ish web applications which use Moose heavily +and have been in production (without issue) for several months now. At +$work, we are re-writing our core offering in it. And several people on +#moose have been using it (in production) for several months now as well. + +Of course, in the end, you need to make this call yourself. If you have +any questions or concerns, please feel free to email me, or even the list +or just stop by #moose and ask away. =head2 Is Moose just Perl 6 in Perl 5? -No. While Moose is very much inspired by Perl 6, it is not. Instead, it -is an OO system for Perl 5. I built Moose because I was tired or writing -the same old boring Perl 5 OO code, and drooling over Perl 6 OO. So -instead of switching to Ruby, I wrote Moose :) +No. While Moose is very much inspired by Perl 6, it is not itself Perl 6. +Instead, it is an OO system for Perl 5. I built Moose because I was tired or +writing the same old boring Perl 5 OO code, and drooling over Perl 6 OO. So +instead of switching to Ruby, I wrote Moose :) =head1 BUILDING CLASSES WITH MOOSE -Moose makes every attempt to provide as much convience during class -construction/definition, but still stay out of your way if you want -it to. Here are a few items to note when building classes with Moose. +Moose makes every attempt to provide as much convenience as possible during +class construction/definition, but still stay out of your way if you want it +to. Here are a few items to note when building classes with Moose. Unless specified with C, any class which uses Moose will inherit from L. Moose will also manage all attributes (including inherited ones) that -are defined with C. And assuming that you call C which is +are defined with C. And assuming that you call C, which is inherited from L, then this includes properly initializing -all instance slots, setting defaults where approprtiate and performing any +all instance slots, setting defaults where appropriate, and performing any type constraint checking or coercion. =head1 EXPORTED FUNCTIONS -Moose will export a number of functions into the class's namespace, which +Moose will export a number of functions into the class's namespace which can then be used to set up the class. These functions all work directly on the current class. @@ -373,14 +367,14 @@ This function will set the superclass(es) for the current class. This approach is recommended instead of C, because C actually Ces onto the class's C<@ISA>, whereas C will replace it. This is important to ensure that classes which do not have -superclasses properly inherit from L. +superclasses still properly inherit from L. =item B This will apply a given set of C<@roles> to the local class. Role support -is currently under heavy development, see L for more details. +is currently under heavy development; see L for more details. -=item B +=item B %options> This will install an attribute of a given C<$name> into the current class. The list of C<%options> are the same as those provided by @@ -403,13 +397,13 @@ I, I and I options inherited from L option uses Moose's type constraint facilities to set up runtime type checking for this attribute. Moose will perform the checks during class construction, and within any accessors. The C<$type_name> argument must be a -string. The string can be either a class name, or a type defined using -Moose's type defintion features. +string. The string can be either a class name or a type defined using +Moose's type definition features. =item I (1|0)> This will attempt to use coercion with the supplied type constraint to change -the value passed into any accessors of constructors. You B have supplied +the value passed into any accessors or constructors. You B have supplied a type constraint in order for this to work. See L for an example usage. @@ -426,31 +420,192 @@ an accessor. =item I (1|0)> -This will tell the class to strore the value of this attribute as a weakened -reference. If an attribute is a weakened reference, it can B also be coerced. +This will tell the class to store the value of this attribute as a weakened +reference. If an attribute is a weakened reference, it B also be +coerced. =item I (1|0)> -This will tell the class to not create this slot until absolutely nessecary. +This will tell the class to not create this slot until absolutely necessary. If an attribute is marked as lazy it B have a default supplied. =item I (1|0)> -This tells the accessor whether to automatically de-reference the value returned. +This tells the accessor whether to automatically dereference the value returned. This is only legal if your C option is either an C or C. +=item I $metaclass_name> + +This tells the class to use a custom attribute metaclass for this particular +attribute. Custom attribute metaclasses are useful for extending the capabilities +of the I keyword, they are the simplest way to extend the MOP, but they are +still a fairly advanced topic and too much to cover here. I will try and write a +recipe on it soon. + +The default behavior here is to just load C<$metaclass_name>, however, we also +have a way to alias to a shorter name. This will first look to see if +B exists, if it does it will +then check to see if that has the method C which +should return the actual name of the custom attribute metaclass. If there is +no C method, it will just default to using +B as the metaclass name. + =item I $code> The trigger option is a CODE reference which will be called after the value of the attribute is set. The CODE ref will be passed the instance itself, the updated value and the attribute meta-object (this is for more advanced fiddling -and can typically be ignored in most cases). You can B have a trigger on +and can typically be ignored in most cases). You B have a trigger on a read-only attribute. -=item I [ @handles ]> +=item I ARRAY | HASH | REGEXP | CODE> + +The handles option provides Moose classes with automated delegation features. +This is a pretty complex and powerful option, it accepts many different option +formats, each with it's own benefits and drawbacks. + +B This features is no longer experimental, but it still may have subtle +bugs lurking in the deeper corners. So if you think you have found a bug, you +probably have, so please report it to me right away. + +B The class being delegated to does not need to be a Moose based class. +Which is why this feature is especially useful when wrapping non-Moose classes. + +All handles option formats share the following traits. + +You cannot override a locally defined method with a delegated method, an +exception will be thrown if you try. Meaning, if you define C in your +class, you cannot override it with a delegated C. This is almost never +something you would want to do, and if it is, you should do it by hand and +not use Moose. + +You cannot override any of the methods found in Moose::Object as well as +C or C methods. These will not throw an exception, but will +silently move on to the next method in the list. My reasoning for this is that +you would almost never want to do this because it usually tends to break your +class. And as with overriding locally defined methods, if you do want to do this, +you should do it manually and not with Moose. + +Below is the documentation for each option format: + +=over 4 + +=item C + +This is the most common usage for handles. You basically pass a list of +method names to be delegated, and Moose will install a delegation method +for each one in the list. + +=item C + +This is the second most common usage for handles. Instead of a list of +method names, you pass a HASH ref where the key is the method name you +want installed locally, and the value is the name of the original method +in the class being delegated to. + +This can be very useful for recursive classes like trees, here is a +quick example (soon to be expanded into a Moose::Cookbook::Recipe): + + pacakge Tree; + use Moose; + + has 'node' => (is => 'rw', isa => 'Any'); + + has 'children' => ( + is => 'ro', + isa => 'ArrayRef', + default => sub { [] } + ); + + has 'parent' => ( + is => 'rw', + isa => 'Tree', + is_weak_ref => 1, + handles => { + parent_node => 'node', + siblings => 'children', + } + ); -There is experimental support for attribute delegation using the C -option. More docs to come later. +In this example, the Tree package gets the C and C methods +which delegate to the C and C methods of the Tree instance stored +in the parent slot. + +=item C + +The regexp option works very similar to the ARRAY option, except that it builds +the list of methods for you. It starts by collecting all possible methods of the +class being delegated to, then filters that list using the regexp supplied here. + +B An I option is required when using the regexp option format. This +is so that we can determine (at compile time) the method list from the class. +Without an I this is just not possible. + +=item C + +This is the option to use when you really want to do something funky. You should +only use it if you really know what you are doing as it involves manual metaclass +twiddling. + +This takes a code reference, which should expect two arguments. The first is +the attribute meta-object this I is attached to. The second is the metaclass +of the class being delegated to. It expects you to return a hash (not a HASH ref) +of the methods you want mapped. + +=back + +=back + +=item B %options> + +This is variation on the normal attibute creator C, which allows you to +clone and extend an attribute from a superclass. Here is a quick example: + + package Foo; + use Moose; + + has 'message' => ( + is => 'rw', + isa => 'Str', + default => 'Hello, I am a Foo' + ); + + package My::Foo; + use Moose; + + extends 'Foo'; + + has '+message' => (default => 'Hello I am My::Foo'); + +What is happening here is that B is cloning the C attribute +from it's parent class B, retaining the is =E 'rw' and isa =E 'Str' +characteristics, but changing the value in C. + +This feature is restricted somewhat, so as to try and enfore at least I +sanity into it. You are only allowed to change the following attributes: + +=over 4 + +=item I + +Change the default value of an attribute. + +=item I + +Change whether the attribute attempts to coerce a value passed to it. + +=item I + +Change if the attribute is required to have a value. + +=item I + +Change the documentation string associated with the attribute. + +=item I + +You I allowed to change the type, but if and B the new type is +a subtype of the old type. =back @@ -460,72 +615,76 @@ option. More docs to come later. =item B sub { ... }> -This three items are syntactic sugar for the before, after and around method +This three items are syntactic sugar for the before, after, and around method modifier features that L provides. More information on these can be found in the L documentation for now. =item B -The keyword C is a noop when called outside of an C method. In +The keyword C is a no-op when called outside of an C method. In the context of an C method, it will call the next most appropriate superclass method with the same arguments as the original method. =item B -An C method, is a way of explictly saying "I am overriding this +An C method is a way of explicitly saying "I am overriding this method from my superclass". You can call C within this method, and it will work as expected. The same thing I be accomplished with a normal -method call and the C pseudo-package, it is really your choice. +method call and the C pseudo-package; it is really your choice. =item B The keyword C, much like C, is a no-op outside of the context of an C method. You can think of C as being the inverse of -C, the details of how C and C work is best described in +C; the details of how C and C work is best described in the L. =item B -An C method, is a way of explictly saying "I am augmenting this +An C method, is a way of explicitly saying "I am augmenting this method from my superclass". Once again, the details of how C and C work is best described in the L. =item B -This is the C function, and exported here beause I use it +This is the C function, and exported here because I use it all the time. This feature may change in the future, so you have been warned. =item B -This is the C function, it is exported here beause I +This is the C function, it is exported here because I use it all the time. It is highly recommended that this is used instead of C anywhere you need to test for an object's class name. =back -=head1 FUTURE PLANS +=head1 UNEXPORTING FUNCTIONS -Here is just a sampling of the plans we have in store for Moose: +=head2 B -=over 4 - -=item * +Moose offers a way of removing the keywords it exports though the C +method. You simply have to say C at the bottom of your code for this +to work. Here is an example: -Compiling Moose classes/roles into C<.pmc> files for faster loading and execution. + package Person; + use Moose; -=item * - -Supporting sealed and finalized classes in Moose. This will allow greater control -of the extensions of frameworks and such. - -=back + has 'first_name' => (is => 'rw', isa => 'Str'); + has 'last_name' => (is => 'rw', isa => 'Str'); + + sub full_name { + my $self = shift; + $self->first_name . ' ' . $self->last_name + } + + no Moose; # keywords are removed from the Person package =head1 MISC. =head2 What does Moose stand for?? Moose doesn't stand for one thing in particular, however, if you -want, here are a few of my favorites, feel free to contribute +want, here are a few of my favorites; feel free to contribute more :) =over 4 @@ -552,17 +711,17 @@ more :) =item * -It should be noted that C and C can B be used in the same -method. However, they can be combined together with the same class hierarchy, +It should be noted that C and C C be used in the same +method. However, they can be combined together with the same class hierarchy; see F for an example. -The reason that this is so is because C is only valid within a method +The reason for this is that C is only valid within a method with the C modifier, and C will never be valid within an C method. In fact, C will skip over any C methods -when searching for it's appropriate C. +when searching for its appropriate C. This might seem like a restriction, but I am of the opinion that keeping these -two features seperate (but interoperable) actually makes them easy to use since +two features separate (but interoperable) actually makes them easy to use, since their behavior is then easier to predict. Time will tell if I am right or not. =back @@ -582,7 +741,9 @@ and it certainly wouldn't have this name ;P originally, I just ran with it. =item Thanks to mst & chansen and the whole #moose poose for all the -ideas/feature-requests/encouragement +ideas/feature-requests/encouragement/bug-finding. + +=item Thanks to David "Theory" Wheeler for meta-discussions and spelling fixes. =back @@ -594,6 +755,8 @@ ideas/feature-requests/encouragement =item The #moose channel on irc.perl.org +=item The Moose mailing list - moose@perl.org + =item L =item L @@ -620,7 +783,7 @@ Yuval Kogman Enothingmuch@woobling.orgE =head1 COPYRIGHT AND LICENSE -Copyright 2006 by Infinity Interactive, Inc. +Copyright 2006, 2007 by Infinity Interactive, Inc. L