1#!/usr/bin/perl
2
3use strict;
4use warnings;
5
6require q(./test.pl); plan(tests => 10);
7
8=pod
9
10This tests the classic diamond inheritance pattern.
11
12   <A>
13  /   \
14<B>   <C>
15  \   /
16   <D>
17
18=cut
19
20{
21    package Diamond_A;
22    use mro 'c3'; 
23    sub bar { 'Diamond_A::bar' }        
24    sub baz { 'Diamond_A::baz' }
25}
26{
27    package Diamond_B;
28    use base 'Diamond_A';
29    use mro 'c3';    
30    sub baz { 'Diamond_B::baz => ' . (shift)->next::method() }         
31}
32{
33    package Diamond_C;
34    use mro 'c3';    
35    use base 'Diamond_A';     
36    sub foo { 'Diamond_C::foo' }   
37    sub buz { 'Diamond_C::buz' }     
38    
39    sub woz { 'Diamond_C::woz' }
40    sub maybe { 'Diamond_C::maybe' }         
41}
42{
43    package Diamond_D;
44    use base ('Diamond_B', 'Diamond_C');
45    use mro 'c3'; 
46    sub foo { 'Diamond_D::foo => ' . (shift)->next::method() } 
47    sub bar { 'Diamond_D::bar => ' . (shift)->next::method() }   
48    sub buz { 'Diamond_D::buz => ' . (shift)->baz() }  
49    sub fuz { 'Diamond_D::fuz => ' . (shift)->next::method() }  
50    
51    sub woz { 'Diamond_D::woz can => ' . ((shift)->next::can() ? 1 : 0) }
52    sub noz { 'Diamond_D::noz can => ' . ((shift)->next::can() ? 1 : 0) }
53
54    sub maybe { 'Diamond_D::maybe => ' . ((shift)->maybe::next::method() || 0) }
55    sub moybe { 'Diamond_D::moybe => ' . ((shift)->maybe::next::method() || 0) }             
56
57}
58
59ok(eq_array(
60    mro::get_linear_isa('Diamond_D'),
61    [ qw(Diamond_D Diamond_B Diamond_C Diamond_A) ]
62), '... got the right MRO for Diamond_D');
63
64is(Diamond_D->foo, 'Diamond_D::foo => Diamond_C::foo', '... skipped B and went to C correctly');
65is(Diamond_D->bar, 'Diamond_D::bar => Diamond_A::bar', '... skipped B & C and went to A correctly');
66is(Diamond_D->baz, 'Diamond_B::baz => Diamond_A::baz', '... called B method, skipped C and went to A correctly');
67is(Diamond_D->buz, 'Diamond_D::buz => Diamond_B::baz => Diamond_A::baz', '... called D method dispatched to , different method correctly');
68eval { Diamond_D->fuz };
69like($@, qr/^No next::method 'fuz' found for Diamond_D/, '... cannot re-dispatch to a method which is not there');
70
71is(Diamond_D->woz, 'Diamond_D::woz can => 1', '... can re-dispatch figured out correctly');
72is(Diamond_D->noz, 'Diamond_D::noz can => 0', '... cannot re-dispatch figured out correctly');
73
74is(Diamond_D->maybe, 'Diamond_D::maybe => Diamond_C::maybe', '... redispatched D to C when it exists');
75is(Diamond_D->moybe, 'Diamond_D::moybe => 0', '... quietly failed redispatch from D');
76