blob: dcee5c518ab87a2f16727f02b96bc6c4c4343f1a [file] [log] [blame]
#!./perl -w
BEGIN {
if ($ENV{PERL_CORE}) {
chdir 't' if -d 't';
#@INC = '../lib';
}
}
use strict;
use File::Spec;
use File::Path;
my $dir;
BEGIN
{
$dir = File::Spec->catdir( "auto-$$" );
unshift @INC, $dir;
}
use Test::More tests => 18;
sub write_file {
my ($file, $text) = @_;
open my $fh, '>', $file
or die "Could not open file '$file' for writing: $!";
print $fh $text;
close $fh;
}
# First we must set up some autoloader files
my $fulldir = File::Spec->catdir( $dir, 'auto', 'Foo' );
mkpath( $fulldir ) or die "Can't mkdir '$fulldir': $!";
write_file( File::Spec->catfile( $fulldir, 'foo.al' ), <<'EOT' );
package Foo;
sub foo { shift; shift || "foo" }
1;
EOT
write_file( File::Spec->catfile( $fulldir, 'bazmarkhian.al' ), <<'EOT' );
package Foo;
sub bazmarkhianish { shift; shift || "baz" }
1;
EOT
my $blechanawilla_text = <<'EOT';
package Foo;
sub blechanawilla { compilation error (
EOT
write_file( File::Spec->catfile( $fulldir, 'blechanawilla.al' ), $blechanawilla_text );
# This is just to keep the old SVR3 systems happy; they may fail
# to find the above file so we duplicate it where they should find it.
write_file( File::Spec->catfile( $fulldir, 'blechanawil.al' ), $blechanawilla_text );
# Let's define the package
package Foo;
require AutoLoader;
AutoLoader->import( 'AUTOLOAD' );
sub new { bless {}, shift };
sub foo;
sub bazmarkhianish;
package main;
my $foo = Foo->new();
my $result = $foo->can( 'foo' );
ok( $result, 'can() first time' );
is( $foo->foo, 'foo', 'autoloaded first time' );
is( $foo->foo, 'foo', 'regular call' );
is( $result, \&Foo::foo, 'can() returns ref to regular installed sub' );
eval {
$foo->will_fail;
};
like( $@, qr/^Can't locate/, 'undefined method' );
$result = $foo->can( 'will_fail' );
ok( ! $result, 'can() should fail on undefined methods' );
# Used to be trouble with this
eval {
my $foo = Foo->new();
die "oops";
};
like( $@, qr/oops/, 'indirect method call' );
# Pass regular expression variable to autoloaded function. This used
# to go wrong because AutoLoader used regular expressions to generate
# autoloaded filename.
'foo' =~ /(\w+)/;
is( $foo->bazmarkhianish($1), 'foo', 'autoloaded method should not stomp match vars' );
is( $foo->bazmarkhianish($1), 'foo', '(again)' );
# Used to retry long subnames with shorter filenames on any old
# exception, including compilation error. Now AutoLoader only
# tries shorter filenames if it can't find the long one.
eval {
$foo->blechanawilla;
};
like( $@, qr/syntax error/i, 'require error propagates' );
# test recursive autoloads
write_file( File::Spec->catfile( $fulldir, 'a.al' ), <<'EOT' );
package Foo;
BEGIN { b() }
sub a { ::ok( 1, 'adding a new autoloaded method' ); }
1;
EOT
write_file( File::Spec->catfile( $fulldir, 'b.al' ), <<'EOT' );
package Foo;
sub b { ::ok( 1, 'adding a new autoloaded method' ) }
1;
EOT
Foo::a();
package Bar;
AutoLoader->import();
::ok( ! defined &AUTOLOAD, 'AutoLoader should not export AUTOLOAD by default' );
::ok( ! defined &can, '... nor can()' );
package Foo;
AutoLoader->unimport();
eval { Foo->baz() };
::like( $@, qr/locate object method "baz"/,
'unimport() should remove imported AUTOLOAD()' );
package Baz;
sub AUTOLOAD { 'i am here' }
AutoLoader->import();
AutoLoader->unimport();
::is( Baz->AUTOLOAD(), 'i am here', '... but not non-imported AUTOLOAD()' );
package SomeClass;
use AutoLoader 'AUTOLOAD';
sub new {
bless {} => shift;
}
package main;
$INC{"SomeClass.pm"} = $0; # Prepare possible recursion
{
my $p = SomeClass->new();
} # <-- deep recursion in AUTOLOAD looking for SomeClass::DESTROY?
::ok(1, "AutoLoader shouldn't loop forever if \%INC is modified");
# Now test the bug that lead to AutoLoader 0.67:
# If the module is loaded from a file name different than normal,
# we could formerly have trouble finding autosplit.ix
# Contributed by Christoph Lamprecht.
# Recreate the following file structure:
# auto/MyAddon/autosplit.ix
# auto/MyAddon/testsub.al
# MyModule.pm
SCOPE: {
my $autopath = File::Spec->catdir( $dir, 'auto', 'MyAddon' );
mkpath( $autopath ) or die "Can't mkdir '$autopath': $!";
my $autosplit_text = <<'EOT';
# Index created by AutoSplit for MyModule.pm
# (file acts as timestamp)
package MyAddon;
sub testsub ;
1;
EOT
write_file( File::Spec->catfile( $autopath, 'autosplit.ix' ), $autosplit_text );
my $testsub_text = <<'EOT';
# NOTE: Derived from MyModule.pm.
# Changes made here will be lost when autosplit is run again.
# See AutoSplit.pm.
package MyAddon;
#line 13 "MyModule.pm (autosplit into auto/MyAddon/testsub.al)"
sub testsub{
return "MyAddon";
}
1;
# end of MyAddon::testsub
EOT
write_file( File::Spec->catfile( $autopath, 'testsub.al' ), $testsub_text);
my $mymodule_text = <<'EOT';
use strict;
use warnings;
package MyModule;
sub testsub{return 'MyModule';}
package MyAddon;
our @ISA = ('MyModule');
BEGIN{$INC{'MyAddon.pm'} = __FILE__}
use AutoLoader 'AUTOLOAD';
1;
__END__
sub testsub{
return "MyAddon";
}
EOT
write_file( File::Spec->catfile( $dir, 'MyModule.pm' ), $mymodule_text);
require MyModule;
my $res = MyAddon->testsub();
::is ($res , 'MyAddon', 'invoke MyAddon::testsub');
}
# cleanup
END {
return unless $dir && -d $dir;
rmtree $dir;
}