1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121
|
use warnings;
use strict;
use Test::More tests => 42;
BEGIN { use_ok "Module::Runtime", qw(use_package_optimistically); }
unshift @INC, "./t/lib";
my $result;
# a module that doesn't exist
$result = eval { use_package_optimistically("t::NotExist") };
is $@, "";
is $result, "t::NotExist";
# a module that's already loaded
$result = eval { use_package_optimistically("Test::More") };
is $@, "";
is $result, "Test::More";
# a module that we'll load now
$result = eval { use_package_optimistically("t::Simple") };
is $@, "";
is $result, "t::Simple";
no strict "refs";
ok defined(${"t::Simple::VERSION"});
# lexical hints don't leak through
my $have_runtime_hint_hash = "$]" >= 5.009004;
sub test_runtime_hint_hash($$) {
SKIP: {
skip "no runtime hint hash", 1 unless $have_runtime_hint_hash;
is +((caller(0))[10] || {})->{$_[0]}, $_[1];
}
}
SKIP: {
skip "core bug makes this test crash", 13
if "$]" >= 5.008 && "$]" < 5.008004;
skip "can't work around hint leakage in pure Perl", 13
if "$]" >= 5.009004 && "$]" < 5.010001;
$^H |= 0x20000 if "$]" < 5.009004;
$^H{"Module::Runtime/test_a"} = 1;
is $^H{"Module::Runtime/test_a"}, 1;
is $^H{"Module::Runtime/test_b"}, undef;
use_package_optimistically("t::Hints");
is $^H{"Module::Runtime/test_a"}, 1;
is $^H{"Module::Runtime/test_b"}, undef;
t::Hints->import;
is $^H{"Module::Runtime/test_a"}, 1;
is $^H{"Module::Runtime/test_b"}, 1;
eval q{
BEGIN { $^H |= 0x20000; $^H{foo} = 1; }
BEGIN { is $^H{foo}, 1; }
main::test_runtime_hint_hash("foo", 1);
BEGIN { use_package_optimistically("Math::BigInt"); }
BEGIN { is $^H{foo}, 1; }
main::test_runtime_hint_hash("foo", 1);
1;
}; die $@ unless $@ eq "";
}
# broken module is visibly broken when re-required
eval { use_package_optimistically("t::Break") };
like $@, qr/\A(?:broken |Attempt to reload )/;
eval { use_package_optimistically("t::Break") };
like $@, qr/\A(?:broken |Attempt to reload )/;
# module broken by virtue of trying to non-optimistically load a
# non-existent module via "use"
eval { use_package_optimistically("t::Nest0") };
like $@, qr/\ACan't locate /;
eval { use_package_optimistically("t::Nest0") };
like $@, qr/\A(?:Can't locate |Attempt to reload )/;
# module broken by virtue of trying to non-optimistically load a
# non-existent module via require_module()
eval { use_package_optimistically("t::Nest1") };
like $@, qr/\ACan't locate /;
eval { use_package_optimistically("t::Nest1") };
like $@, qr/\A(?:Can't locate |Attempt to reload )/;
# successful version check
$result = eval { use_package_optimistically("Module::Runtime", 0.001) };
is $@, "";
is $result, "Module::Runtime";
# failing version check
$result = eval { use_package_optimistically("Module::Runtime", 999) };
like $@, qr/^Module::Runtime version /;
# even load module if $VERSION already set, unlike older behaviour
$t::Context::VERSION = undef;
$result = eval { use_package_optimistically("t::Context") };
is $@, "";
is $result, "t::Context";
ok defined($t::Context::VERSION);
ok $INC{"t/Context.pm"};
# make sure any version argument gets passed through
my @version_calls;
sub t::HasVersion::VERSION {
push @version_calls, [@_];
}
$INC{"t/HasVersion.pm"} = 1;
eval { use_package_optimistically("t::HasVersion") };
is $@, "";
is_deeply \@version_calls, [];
@version_calls = ();
eval { use_package_optimistically("t::HasVersion", 2) };
is $@, "";
is_deeply \@version_calls, [["t::HasVersion",2]];
@version_calls = ();
eval { use_package_optimistically("t::HasVersion", "wibble") };
is $@, "";
is_deeply \@version_calls, [["t::HasVersion","wibble"]];
@version_calls = ();
eval { use_package_optimistically("t::HasVersion", undef) };
is $@, "";
is_deeply \@version_calls, [["t::HasVersion",undef]];
1;
|