From Code to Community: Sponsoring The Perl and Raku Conference 2025 Learn more

#!/usr/bin/perl
use strict;
BEGIN { chdir 't' if -d 't'; require q(./test.pl);
set_up_inc('../lib', 'lib');
}
plan(tests => 12);
{
{
package Foo;
use strict;
use warnings;
use mro 'c3';
sub new { bless {}, $_[0] }
sub bar { 'Foo::bar' }
}
# call the submethod in the direct instance
my $foo = Foo->new();
object_ok($foo, 'Foo');
can_ok($foo, 'bar');
is($foo->bar(), 'Foo::bar', '... got the right return value');
# fail calling it from a subclass
{
package Bar;
use strict;
use warnings;
use mro 'c3';
our @ISA = ('Foo');
}
my $bar = Bar->new();
object_ok($bar, 'Bar');
object_ok($bar, 'Foo');
# test it working with Sub::Name
SKIP: {
eval 'use Sub::Name';
skip("Sub::Name is required for this test", 3) if $@;
my $m = sub { (shift)->next::method() };
Sub::Name::subname('Bar::bar', $m);
{
no strict 'refs';
*{'Bar::bar'} = $m;
}
can_ok($bar, 'bar');
my $value = eval { $bar->bar() };
ok(!$@, '... calling bar() succeeded') || diag $@;
is($value, 'Foo::bar', '... got the right return value too');
}
# test it failing without Sub::Name
{
package Baz;
use strict;
use warnings;
use mro 'c3';
our @ISA = ('Foo');
}
my $baz = Baz->new();
object_ok($baz, 'Baz');
object_ok($baz, 'Foo');
{
my $m = sub { (shift)->next::method() };
{
no strict 'refs';
*{'Baz::bar'} = $m;
}
eval { $baz->bar() };
ok($@, '... calling bar() with next::method failed') || diag $@;
}
# Test with non-existing class (used to segfault)
{
package Qux;
use mro;
sub foo { No::Such::Class->next::can }
}
eval { Qux->foo() };
is($@, '', "->next::can on non-existing package name");
}