0.41
[gitmo/Class-MOP.git] / t / 003_methods.t
1 #!/usr/bin/perl
2
3 use strict;
4 use warnings;
5
6 use Test::More tests => 66;
7 use Test::Exception;
8
9 use Scalar::Util qw/reftype/;
10
11 BEGIN {
12     use_ok('Class::MOP');   
13     use_ok('Class::MOP::Class');        
14 }
15
16 {   # This package tries to test &has_method 
17     # as exhaustively as possible. More corner
18     # cases are welcome :)
19     package Foo;
20     
21     # import a sub
22     use Scalar::Util 'blessed'; 
23     
24     sub pie;
25     sub cake ();
26
27     use constant FOO_CONSTANT => 'Foo-CONSTANT';
28     
29     # define a sub in package
30     sub bar { 'Foo::bar' } 
31     *baz = \&bar;
32
33     { # method named with Sub::Name inside the package scope
34         no strict 'refs';
35         *{'Foo::floob'} = Sub::Name::subname 'floob' => sub { '!floob!' }; 
36     }
37
38     # We hateses the "used only once" warnings
39     { my $temp = \&Foo::baz }
40     
41     package OinkyBoinky;
42     our @ISA = "Foo";
43     
44     sub elk { 'OinkyBoinky::elk' }
45
46     package main;
47     
48     sub Foo::blah { $_[0]->Foo::baz() }
49     
50     {
51         no strict 'refs';
52         *{'Foo::bling'} = sub { '$$Bling$$' };
53         *{'Foo::bang'} = Sub::Name::subname 'Foo::bang' => sub { '!BANG!' }; 
54         *{'Foo::boom'} = Sub::Name::subname 'boom' => sub { '!BOOM!' };     
55         
56         eval "package Foo; sub evaled_foo { 'Foo::evaled_foo' }";           
57     }
58 }
59
60 my $Foo = Class::MOP::Class->initialize('Foo');
61
62 ok(!$Foo->has_method('pie'), '... got the method stub pie');
63 ok(!$Foo->has_method('cake'), '... got the constant method stub cake');
64
65 my $foo = sub { 'Foo::foo' };
66
67 ok(!UNIVERSAL::isa($foo, 'Class::MOP::Method'), '... our method is not yet blessed');
68
69 lives_ok {
70     $Foo->add_method('foo' => $foo);
71 } '... we added the method successfully';
72
73 my $foo_method = $Foo->get_method('foo');
74
75 isa_ok($foo_method, 'Class::MOP::Method');
76
77 is($foo_method->name, 'foo', '... got the right name for the method');
78 is($foo_method->package_name, 'Foo', '... got the right package name for the method');
79
80 ok($Foo->has_method('foo'), '... Foo->has_method(foo) (defined with Sub::Name)');
81
82 is($Foo->get_method('foo')->body, $foo, '... Foo->get_method(foo) == \&foo');
83 is(Foo->foo(), 'Foo::foo', '... Foo->foo() returns "Foo::foo"');
84
85 # now check all our other items ...
86
87 if ($Foo->has_method('FOO_CONSTANT')) {
88     pass('... Foo->has_method(FOO_CONSTANT) (defined w/ use constant)');
89 }
90 else {
91     diag(q{
92         FIXME:
93         You are using bleadperl or 5.9.5 which handles constants 
94         in a differnt way then prior versions of perl. This will
95         cause this test to break, but the test it not critical 
96         to the operation of this module, so I am letting pass 
97         with a big FIXME note until I have the tuits to install
98         5.9.5 and fix it. 
99         
100         Of course, patches are *always* welcome :) });    
101     pass('... FIXME: Foo->has_method(FOO_CONSTANT) (defined w/ use constant)');    
102 }
103 ok($Foo->has_method('bar'), '... Foo->has_method(bar) (defined in Foo)');
104 ok($Foo->has_method('baz'), '... Foo->has_method(baz) (typeglob aliased within Foo)');
105 ok($Foo->has_method('floob'), '... Foo->has_method(floob) (defined in Foo:: using symbol tables and Sub::Name w/out package name)');
106 ok($Foo->has_method('blah'), '... Foo->has_method(blah) (defined in main:: using fully qualified package name)');
107 ok($Foo->has_method('bling'), '... Foo->has_method(bling) (defined in main:: using symbol tables (no Sub::Name))');
108 ok($Foo->has_method('bang'), '... Foo->has_method(bang) (defined in main:: using symbol tables and Sub::Name)');
109 ok($Foo->has_method('evaled_foo'), '... Foo->has_method(evaled_foo) (evaled in main::)');
110
111 my $OinkyBoinky = Class::MOP::Class->initialize('OinkyBoinky');
112
113 ok($OinkyBoinky->has_method('elk'), "the method 'elk' is defined in OinkyBoinky");
114
115 ok(!$OinkyBoinky->has_method('bar'), "the method 'bar' is not defined in OinkyBoinky");
116
117 ok(my $bar = $OinkyBoinky->find_method_by_name('bar'), "but if you look in the inheritence chain then 'bar' does exist");
118
119 is( reftype($bar->body), "CODE", "the returned value is a code ref" );
120
121
122 # calling get_method blessed them all
123 for my $method_name (qw/FOO_CONSTANT
124                         bar
125                         baz
126                         floob
127                         blah            
128                         bling
129                         bang    
130                         evaled_foo/) {
131     isa_ok($Foo->get_method($method_name), 'Class::MOP::Method');
132     {
133         no strict 'refs';
134         is($Foo->get_method($method_name)->body, \&{'Foo::' . $method_name}, '... body matches CODE ref in package for ' . $method_name);
135     }
136 }
137
138 {
139     package Foo::Aliasing;
140     use metaclass;
141     sub alias_me { '...' }
142 }
143
144 $Foo->alias_method('alias_me' => Foo::Aliasing->meta->get_method('alias_me'));
145
146 ok(!$Foo->has_method('alias_me'), '... !Foo->has_method(alias_me) (aliased from Foo::Aliasing)');
147 ok(defined &Foo::alias_me, '... Foo does have a symbol table slow for alias_me though');
148
149 ok(!$Foo->has_method('blessed'), '... !Foo->has_method(blessed) (imported into Foo)');
150 ok(!$Foo->has_method('boom'), '... !Foo->has_method(boom) (defined in main:: using symbol tables and Sub::Name w/out package name)');
151
152 ok(!$Foo->has_method('not_a_real_method'), '... !Foo->has_method(not_a_real_method) (does not exist)');
153 is($Foo->get_method('not_a_real_method'), undef, '... Foo->get_method(not_a_real_method) == undef');
154
155 is_deeply(
156     [ sort $Foo->get_method_list ],
157     [ qw(FOO_CONSTANT bang bar baz blah bling evaled_foo floob foo) ],
158     '... got the right method list for Foo');
159
160 is_deeply(
161     [ sort { $a->{name} cmp $b->{name} } $Foo->compute_all_applicable_methods() ],
162     [
163         map {
164             {
165             name  => $_,
166             class => 'Foo',
167             code  => $Foo->get_method($_)
168             }
169         } qw(
170             FOO_CONSTANT
171             bang 
172             bar 
173             baz 
174             blah 
175             bling 
176             evaled_foo 
177             floob 
178             foo
179         )
180     ],
181     '... got the right list of applicable methods for Foo');
182
183 is($Foo->remove_method('foo')->body, $foo, '... removed the foo method');
184 ok(!$Foo->has_method('foo'), '... !Foo->has_method(foo) we just removed it');
185 dies_ok { Foo->foo } '... cannot call Foo->foo because it is not there';
186
187 is_deeply(
188     [ sort $Foo->get_method_list ],
189     [ qw(FOO_CONSTANT bang bar baz blah bling evaled_foo floob) ],
190     '... got the right method list for Foo');
191
192 ok($Foo->remove_method('FOO_CONSTANT'), '... removed the FOO_CONSTANT method');
193 ok(!$Foo->has_method('FOO_CONSTANT'), '... !Foo->has_method(FOO_CONSTANT) we just removed it');
194 dies_ok { Foo->FOO_CONSTANT } '... cannot call Foo->FOO_CONSTANT because it is not there';
195
196 is_deeply(
197     [ sort $Foo->get_method_list ],
198     [ qw(bang bar baz blah bling evaled_foo floob) ],
199     '... got the right method list for Foo');
200
201 # ... test our class creator 
202
203 my $Bar = Class::MOP::Class->create(
204             'Bar' => (
205                 superclasses => [ 'Foo' ],
206                 methods => {
207                     foo => sub { 'Bar::foo' },
208                     bar => sub { 'Bar::bar' },                    
209                 }
210             ));
211 isa_ok($Bar, 'Class::MOP::Class');
212
213 ok($Bar->has_method('foo'), '... Bar->has_method(foo)');
214 ok($Bar->has_method('bar'), '... Bar->has_method(bar)');
215
216 is(Bar->foo, 'Bar::foo', '... Bar->foo == Bar::foo');
217 is(Bar->bar, 'Bar::bar', '... Bar->bar == Bar::bar');
218
219 lives_ok {
220     $Bar->add_method('foo' => sub { 'Bar::foo v2' });
221 } '... overwriting a method is fine';
222
223 ok($Bar->has_method('foo'), '... Bar-> (still) has_method(foo)');
224 is(Bar->foo, 'Bar::foo v2', '... Bar->foo == "Bar::foo v2"');
225
226 is_deeply(
227     [ sort $Bar->get_method_list ],
228     [ qw(bar foo meta) ],
229     '... got the right method list for Bar');  
230     
231 is_deeply(
232     [ sort { $a->{name} cmp $b->{name} } $Bar->compute_all_applicable_methods() ],
233     [
234         {
235             name  => 'bang',
236             class => 'Foo',
237             code  => $Foo->get_method('bang')
238         },
239         {
240             name  => 'bar',
241             class => 'Bar',
242             code  => $Bar->get_method('bar') 
243         },
244         (map {
245             {
246                 name  => $_,
247                 class => 'Foo',
248                 code  => $Foo->get_method($_)
249             }
250         } qw(        
251             baz 
252             blah 
253             bling 
254             evaled_foo 
255             floob 
256         )),
257         {
258             name  => 'foo',
259             class => 'Bar',
260             code  => $Bar->get_method('foo')
261         },        
262         {
263             name  => 'meta',
264             class => 'Bar',
265             code  => $Bar->get_method('meta')
266         }        
267     ],
268     '... got the right list of applicable methods for Bar');
269
270