move 80x tests to 800_shikabased
[gitmo/Mouse.git] / lib / Mouse / Meta / Role.pm
CommitLineData
a2227e71 1#!/usr/bin/env perl
2package Mouse::Meta::Role;
3use strict;
4use warnings;
59089ec3 5use Carp 'confess';
a2227e71 6
acf0f643 7do {
8 my %METACLASS_CACHE;
9
10 # because Mouse doesn't introspect existing classes, we're forced to
11 # only pay attention to other Mouse classes
12 sub _metaclass_cache {
13 my $class = shift;
14 my $name = shift;
15 return $METACLASS_CACHE{$name};
16 }
17
18 sub initialize {
19 my $class = shift;
20 my $name = shift;
21 $METACLASS_CACHE{$name} = $class->new(name => $name)
22 if !exists($METACLASS_CACHE{$name});
23 return $METACLASS_CACHE{$name};
24 }
25};
26
27sub new {
28 my $class = shift;
29 my %args = @_;
30
59089ec3 31 $args{attributes} ||= {};
32 $args{required_methods} ||= [];
274b6cce 33
acf0f643 34 bless \%args, $class;
35}
a2227e71 36
513854c7 37sub name { $_[0]->{name} }
38
59089ec3 39sub add_required_methods {
40 my $self = shift;
41 my @methods = @_;
42 push @{$self->{required_methods}}, @methods;
43}
44
274b6cce 45sub add_attribute {
46 my $self = shift;
47 my $name = shift;
69ac1dcf 48 my $spec = shift;
49 $self->{attributes}->{$name} = $spec;
da0c885d 50}
51
274b6cce 52sub has_attribute { exists $_[0]->{attributes}->{$_[1]} }
53sub get_attribute_list { keys %{ $_[0]->{attributes} } }
69ac1dcf 54sub get_attribute { $_[0]->{attributes}->{$_[1]} }
274b6cce 55
2e92bb89 56# copied from Class::Inspector
57sub get_method_list {
58 my $self = shift;
59 my $name = $self->name;
60
61 no strict 'refs';
62 # Get all the CODE symbol table entries
63 my @functions = grep !/^meta$/,
64 grep { /\A[^\W\d]\w*\z/o }
65 grep { defined &{"${name}::$_"} }
66 keys %{"${name}::"};
67 wantarray ? @functions : \@functions;
68}
69
da0c885d 70sub apply {
71 my $self = shift;
2e92bb89 72 my $selfname = $self->name;
da0c885d 73 my $class = shift;
2e92bb89 74 my $classname = $class->name;
4aaa2ed6 75 my %args = @_;
da0c885d 76
4aaa2ed6 77 if ($class->isa('Mouse::Meta::Class')) {
78 for my $name (@{$self->{required_methods}}) {
79 unless ($classname->can($name)) {
80 confess "'$selfname' requires the method '$name' to be implemented by '$classname'";
81 }
2e92bb89 82 }
83 }
84
85 {
86 no strict 'refs';
87 for my $name ($self->get_method_list) {
88 next if $name eq 'has' || $name eq 'requires' || $name eq 'meta' || $name eq 'with' || $name eq 'around' || $name eq 'before' || $name eq 'after' || $name eq 'blessed' || $name eq 'extends' || $name eq 'confess' || $name eq 'excludes';
4aaa2ed6 89 my $dstname = $args{alias} ? ($args{alias}->{$name}||$name) : $name;
90 if ($classname->can($dstname)) {
2e92bb89 91 # XXX what's Moose's behavior?
92 next;
93 }
4aaa2ed6 94 *{"${classname}::${dstname}"} = *{"${selfname}::${name}"};
59089ec3 95 }
96 }
97
b1b81553 98 if ($class->isa('Mouse::Meta::Class')) {
99 # apply role to class
100 for my $name ($self->get_attribute_list) {
101 next if $class->has_attribute($name);
102 my $spec = $self->get_attribute($name);
103 Mouse::Meta::Attribute->create($class, $name, %$spec);
104 }
105 } else {
106 # apply role to role
107 # XXX Room for speed improvement
108 for my $name ($self->get_attribute_list) {
109 next if $class->has_attribute($name);
110 my $spec = $self->get_attribute($name);
111 $class->add_attribute($name, $spec);
112 }
da0c885d 113 }
d99db7b6 114
b1b81553 115 # XXX Room for speed improvement in role to role
d99db7b6 116 for my $modifier_type (qw/before after around/) {
117 my $add_method = "add_${modifier_type}_method_modifier";
118 my $modified = $self->{"${modifier_type}_method_modifiers"};
119
120 for my $method_name (keys %$modified) {
121 for my $code (@{ $modified->{$method_name} }) {
122 $class->$add_method($method_name => $code);
123 }
124 }
125 }
da0c885d 126}
0fc8adbc 127
c2f128e7 128for my $modifier_type (qw/before after around/) {
129 no strict 'refs';
fc0e0bbd 130 *{ __PACKAGE__ . '::' . "add_${modifier_type}_method_modifier" } = sub {
131 my ($self, $method_name, $method) = @_;
132
133 push @{ $self->{"${modifier_type}_method_modifiers"}->{$method_name} },
134 $method;
135 };
136
c2f128e7 137 *{ __PACKAGE__ . '::' . "get_${modifier_type}_method_modifiers" } = sub {
138 my ($self, $method_name, $method) = @_;
139 @{ $self->{"${modifier_type}_method_modifiers"}->{$method_name} || [] }
140 };
141}
142
a2227e71 1431;
144